@switchbot/homebridge-switchbot 5.0.0-beta.98 → 5.0.0
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/.changeset/config.json +14 -0
- package/.github/copilot-instructions.md +39 -0
- package/.github/workflows/ci.yml +4 -1
- package/.github/workflows/manual-e2e.yml +6 -3
- package/.github/workflows/release.yml +64 -15
- package/.github/workflows/stale.yml +2 -4
- package/.husky/pre-push +15 -0
- package/CHANGELOG.md +126 -134
- package/MIGRATION.md +16 -6
- package/README.md +84 -3
- package/TODO.md +263 -0
- package/config.schema.json +229 -36
- package/dist/SwitchBotHAPPlatform.d.ts +133 -0
- package/dist/SwitchBotHAPPlatform.d.ts.map +1 -0
- package/dist/SwitchBotHAPPlatform.js +555 -0
- package/dist/SwitchBotHAPPlatform.js.map +1 -0
- package/dist/SwitchBotMatterPlatform.d.ts +141 -0
- package/dist/SwitchBotMatterPlatform.d.ts.map +1 -0
- package/dist/SwitchBotMatterPlatform.js +536 -0
- package/dist/SwitchBotMatterPlatform.js.map +1 -0
- package/dist/device-types.d.ts +31 -0
- package/dist/device-types.d.ts.map +1 -0
- package/dist/device-types.js +246 -0
- package/dist/device-types.js.map +1 -0
- package/dist/deviceCommandMapper.d.ts +10 -0
- package/dist/deviceCommandMapper.d.ts.map +1 -0
- package/dist/deviceCommandMapper.js +319 -0
- package/dist/deviceCommandMapper.js.map +1 -0
- package/dist/deviceFactory.d.ts +3 -2
- package/dist/deviceFactory.d.ts.map +1 -1
- package/dist/deviceFactory.js +107 -29
- package/dist/deviceFactory.js.map +1 -1
- package/dist/devices/genericDevice.d.ts +59 -37
- package/dist/devices/genericDevice.d.ts.map +1 -1
- package/dist/devices/genericDevice.js +376 -78
- package/dist/devices/genericDevice.js.map +1 -1
- package/dist/errors.d.ts +38 -0
- package/dist/errors.d.ts.map +1 -0
- package/dist/errors.js +32 -0
- package/dist/errors.js.map +1 -0
- 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/config.d.ts +3 -0
- package/dist/homebridge-ui/endpoints/config.d.ts.map +1 -0
- package/dist/homebridge-ui/endpoints/config.js +90 -0
- package/dist/homebridge-ui/endpoints/config.js.map +1 -0
- package/dist/homebridge-ui/endpoints/devices.d.ts +6 -0
- package/dist/homebridge-ui/endpoints/devices.d.ts.map +1 -0
- package/dist/homebridge-ui/endpoints/devices.js +144 -0
- package/dist/homebridge-ui/endpoints/devices.js.map +1 -0
- package/dist/homebridge-ui/endpoints/discovery.d.ts +7 -0
- package/dist/homebridge-ui/endpoints/discovery.d.ts.map +1 -0
- package/dist/homebridge-ui/endpoints/discovery.js +219 -0
- package/dist/homebridge-ui/endpoints/discovery.js.map +1 -0
- 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/css/styles.css +483 -0
- package/dist/homebridge-ui/public/index.html +197 -621
- package/dist/homebridge-ui/public/js/advanced-settings.d.ts +3 -0
- package/dist/homebridge-ui/public/js/advanced-settings.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/advanced-settings.js +95 -0
- package/dist/homebridge-ui/public/js/advanced-settings.js.map +1 -0
- package/dist/homebridge-ui/public/js/advanced-settings.ts +94 -0
- package/dist/homebridge-ui/public/js/api.d.ts +66 -0
- package/dist/homebridge-ui/public/js/api.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/api.js +295 -0
- package/dist/homebridge-ui/public/js/api.js.map +1 -0
- package/dist/homebridge-ui/public/js/api.ts +355 -0
- package/dist/homebridge-ui/public/js/app.d.ts +2 -0
- package/dist/homebridge-ui/public/js/app.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/app.js +3722 -0
- package/dist/homebridge-ui/public/js/app.js.map +7 -0
- package/dist/homebridge-ui/public/js/app.ts +22 -0
- package/dist/homebridge-ui/public/js/constants.d.ts +2 -0
- package/dist/homebridge-ui/public/js/constants.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/constants.js +2 -0
- package/dist/homebridge-ui/public/js/constants.js.map +1 -0
- package/dist/homebridge-ui/public/js/constants.ts +1 -0
- package/dist/homebridge-ui/public/js/credentials.d.ts +3 -0
- package/dist/homebridge-ui/public/js/credentials.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/credentials.js +99 -0
- package/dist/homebridge-ui/public/js/credentials.js.map +1 -0
- package/dist/homebridge-ui/public/js/credentials.ts +105 -0
- package/dist/homebridge-ui/public/js/devices-delete.d.ts +3 -0
- package/dist/homebridge-ui/public/js/devices-delete.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/devices-delete.js +199 -0
- package/dist/homebridge-ui/public/js/devices-delete.js.map +1 -0
- package/dist/homebridge-ui/public/js/devices-delete.ts +227 -0
- package/dist/homebridge-ui/public/js/devices.d.ts +9 -0
- package/dist/homebridge-ui/public/js/devices.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/devices.js +98 -0
- package/dist/homebridge-ui/public/js/devices.js.map +1 -0
- package/dist/homebridge-ui/public/js/devices.ts +106 -0
- package/dist/homebridge-ui/public/js/discovery.d.ts +9 -0
- package/dist/homebridge-ui/public/js/discovery.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/discovery.js +1201 -0
- package/dist/homebridge-ui/public/js/discovery.js.map +1 -0
- package/dist/homebridge-ui/public/js/discovery.ts +1335 -0
- package/dist/homebridge-ui/public/js/logger.d.ts +7 -0
- package/dist/homebridge-ui/public/js/logger.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/logger.js +17 -0
- package/dist/homebridge-ui/public/js/logger.js.map +1 -0
- package/dist/homebridge-ui/public/js/logger.ts +17 -0
- package/dist/homebridge-ui/public/js/modal.d.ts +5 -0
- package/dist/homebridge-ui/public/js/modal.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/modal.js +35 -0
- package/dist/homebridge-ui/public/js/modal.js.map +1 -0
- package/dist/homebridge-ui/public/js/modal.ts +35 -0
- package/dist/homebridge-ui/public/js/modals.d.ts +15 -0
- package/dist/homebridge-ui/public/js/modals.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/modals.js +675 -0
- package/dist/homebridge-ui/public/js/modals.js.map +1 -0
- package/dist/homebridge-ui/public/js/modals.ts +765 -0
- package/dist/homebridge-ui/public/js/render.d.ts +71 -0
- package/dist/homebridge-ui/public/js/render.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/render.js +960 -0
- package/dist/homebridge-ui/public/js/render.js.map +1 -0
- package/dist/homebridge-ui/public/js/render.ts +1084 -0
- package/dist/homebridge-ui/public/js/toast.d.ts +6 -0
- package/dist/homebridge-ui/public/js/toast.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/toast.js +38 -0
- package/dist/homebridge-ui/public/js/toast.js.map +1 -0
- package/dist/homebridge-ui/public/js/toast.ts +44 -0
- package/dist/homebridge-ui/public/js/types.d.ts +23 -0
- package/dist/homebridge-ui/public/js/types.d.ts.map +1 -0
- package/dist/homebridge-ui/public/js/types.js +2 -0
- package/dist/homebridge-ui/public/js/types.js.map +1 -0
- package/dist/homebridge-ui/public/js/types.ts +26 -0
- package/dist/homebridge-ui/server.d.ts +1 -3
- package/dist/homebridge-ui/server.d.ts.map +1 -1
- package/dist/homebridge-ui/server.js +8 -450
- package/dist/homebridge-ui/server.js.map +1 -1
- 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 +39 -0
- package/dist/homebridge-ui/utils/config-parser.d.ts.map +1 -0
- package/dist/homebridge-ui/utils/config-parser.js +108 -0
- package/dist/homebridge-ui/utils/config-parser.js.map +1 -0
- package/dist/homebridge-ui/utils/device-migration.d.ts +35 -0
- package/dist/homebridge-ui/utils/device-migration.d.ts.map +1 -0
- package/dist/homebridge-ui/utils/device-migration.js +111 -0
- package/dist/homebridge-ui/utils/device-migration.js.map +1 -0
- package/dist/homebridge-ui/utils/logger.d.ts +7 -0
- package/dist/homebridge-ui/utils/logger.d.ts.map +1 -0
- package/dist/homebridge-ui/utils/logger.js +17 -0
- package/dist/homebridge-ui/utils/logger.js.map +1 -0
- package/dist/index.d.ts +10 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +12 -2
- package/dist/index.js.map +1 -1
- package/dist/settings.d.ts +1 -0
- package/dist/settings.d.ts.map +1 -1
- package/dist/settings.js +1 -0
- package/dist/settings.js.map +1 -1
- package/dist/switchbotClient.d.ts +12 -10
- package/dist/switchbotClient.d.ts.map +1 -1
- package/dist/switchbotClient.js +156 -103
- package/dist/switchbotClient.js.map +1 -1
- package/dist/utils.d.ts +76 -1
- package/dist/utils.d.ts.map +1 -1
- package/dist/utils.js +1121 -4
- package/dist/utils.js.map +1 -1
- package/docs/assets/highlight.css +16 -2
- package/docs/assets/main.js +1 -1
- package/docs/index.html +82 -5
- package/docs/variables/default.html +3 -1
- package/eslint.config.js +9 -5
- package/nodemon.json +2 -2
- package/package.json +34 -21
- package/scripts/build-ui.js +37 -0
- package/scripts/free-dev-ports.mjs +105 -0
- package/scripts/generate-matter-maps.js +34 -17
- package/scripts/sync-device-types.mjs +31 -0
- package/src/SwitchBotHAPPlatform.ts +558 -0
- package/src/SwitchBotMatterPlatform.ts +538 -0
- package/src/device-types.js +246 -0
- package/src/device-types.js.map +1 -0
- package/src/device-types.ts +261 -0
- package/src/deviceCommandMapper.js +319 -0
- package/src/deviceCommandMapper.js.map +1 -0
- package/src/deviceCommandMapper.ts +333 -0
- package/src/deviceFactory.ts +125 -45
- package/src/devices/genericDevice.ts +411 -69
- package/src/errors.js +32 -0
- package/src/errors.js.map +1 -0
- package/src/errors.ts +35 -0
- package/src/homebridge-ui/endpoints/config.ts +110 -0
- package/src/homebridge-ui/endpoints/devices.ts +153 -0
- package/src/homebridge-ui/endpoints/discovery.ts +240 -0
- package/src/homebridge-ui/public/css/styles.css +483 -0
- package/src/homebridge-ui/public/index.html +197 -621
- package/src/homebridge-ui/public/js/advanced-settings.ts +94 -0
- package/src/homebridge-ui/public/js/api.ts +355 -0
- package/src/homebridge-ui/public/js/app.ts +22 -0
- package/src/homebridge-ui/public/js/constants.ts +1 -0
- package/src/homebridge-ui/public/js/credentials.ts +105 -0
- package/src/homebridge-ui/public/js/devices-delete.ts +227 -0
- package/src/homebridge-ui/public/js/devices.ts +106 -0
- package/src/homebridge-ui/public/js/discovery.ts +1335 -0
- package/src/homebridge-ui/public/js/logger.ts +17 -0
- package/src/homebridge-ui/public/js/modal.ts +35 -0
- package/src/homebridge-ui/public/js/modals.ts +765 -0
- package/src/homebridge-ui/public/js/render.ts +1084 -0
- package/src/homebridge-ui/public/js/toast.ts +44 -0
- package/src/homebridge-ui/public/js/types.ts +26 -0
- package/src/homebridge-ui/server.ts +9 -526
- package/src/homebridge-ui/utils/config-parser.ts +125 -0
- package/src/homebridge-ui/utils/device-migration.ts +144 -0
- package/src/homebridge-ui/utils/logger.ts +17 -0
- package/src/index.ts +12 -2
- package/src/settings.js +8 -0
- package/src/settings.js.map +1 -0
- package/src/settings.ts +2 -0
- package/src/switchbotClient.js +247 -0
- package/src/switchbotClient.js.map +1 -0
- package/src/switchbotClient.ts +177 -114
- package/src/utils.ts +1133 -5
- package/test/client/switchbot-client-debounce.spec.ts +35 -0
- package/test/client/switchbot-client-openapi.spec.ts +19 -0
- package/test/client/switchbotClient.spec.ts +64 -0
- package/test/device/device-mapping.spec.ts +23 -0
- package/test/device/deviceBase.spec.ts +26 -0
- package/test/device/deviceFactory-edge.spec.ts +15 -0
- package/test/device/deviceFactory.spec.ts +33 -0
- package/test/device/fan-swing.spec.ts +34 -0
- package/test/device/genericDevice-blepoll.spec.ts +47 -0
- package/test/device/irdevice.spec.ts +9 -0
- package/test/device/lock-users.spec.ts +35 -0
- package/test/device/matter-descriptors.spec.ts +22 -0
- package/test/device/matter-device-state.spec.ts +37 -0
- package/test/e2e/run-e2e.spec.ts +18 -19
- package/test/errors/errors.spec.ts +10 -0
- package/test/helpers/matter-harness.ts +20 -9
- package/test/homebridge-ui/server.spec.ts +9 -0
- package/test/platform/accessory-restore.spec.ts +37 -0
- package/test/platform/matter-childbridge.spec.ts +34 -0
- package/test/platform/matter-integration.spec.ts +33 -0
- package/test/platform/platform-edge.spec.ts +73 -0
- package/test/platform/platform.integration.spec.ts +34 -0
- package/test/utils/utils-extra.spec.ts +10 -0
- package/test/utils/utils.spec.ts +53 -0
- package/todo/TODO.md +80 -0
- package/tsconfig.ui.json +11 -0
- package/.github/npm-version-script-esm.js +0 -97
- package/.github/workflows/beta-release.yml +0 -52
- package/dist/platform.d.ts +0 -35
- package/dist/platform.d.ts.map +0 -1
- package/dist/platform.js +0 -850
- package/dist/platform.js.map +0 -1
- package/src/platform.ts +0 -867
- package/test/accessory-restore.spec.ts +0 -73
- package/test/device-mapping.spec.ts +0 -37
- package/test/deviceFactory.spec.ts +0 -18
- package/test/fan-swing.spec.ts +0 -29
- package/test/lock-users.spec.ts +0 -44
- package/test/matter-childbridge.spec.ts +0 -55
- package/test/matter-descriptors.spec.ts +0 -97
- package/test/matter-device-state.spec.ts +0 -101
- package/test/matter-integration.spec.ts +0 -70
- package/test/platform.integration.spec.ts +0 -55
- package/test/switchbot-client-debounce.spec.ts +0 -131
- package/test/switchbot-client-openapi.spec.ts +0 -56
- package/test/switchbotClient.spec.ts +0 -10
- package/test/utils.spec.ts +0 -20
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../../../src/homebridge-ui/public/js/logger.ts", "../../../../src/homebridge-ui/public/js/types.ts", "../../../../src/homebridge-ui/public/js/modal.ts", "../../../../src/homebridge-ui/public/js/toast.ts", "../../../../src/device-types.ts", "../../../../src/homebridge-ui/public/js/api.ts", "../../../../src/homebridge-ui/public/js/discovery.ts", "../../../../src/homebridge-ui/public/js/constants.ts", "../../../../src/homebridge-ui/public/js/modals.ts", "../../../../src/homebridge-ui/public/js/devices-delete.ts", "../../../../src/homebridge-ui/public/js/render.ts", "../../../../src/homebridge-ui/public/js/devices.ts", "../../../../src/homebridge-ui/public/js/credentials.ts", "../../../../src/homebridge-ui/public/js/app.ts"],
|
|
4
|
+
"sourcesContent": ["/* eslint-disable no-console */\nconst PREFIX = '[SwitchBot UI/html]'\n\nexport const uiLog = {\n info: (message: string, ...parameters: any[]) => {\n console.log(PREFIX, message, ...parameters)\n },\n warn: (message: string, ...parameters: any[]) => {\n console.warn(PREFIX, message, ...parameters)\n },\n error: (message: string, ...parameters: any[]) => {\n console.error(PREFIX, message, ...parameters)\n },\n debug: (message: string, ...parameters: any[]) => {\n console.debug(PREFIX, message, ...parameters)\n },\n}\n", "// Global homebridge API type declarations\ndeclare global {\n interface HomebridgeToastApi {\n success?: (message: string, title?: string) => void\n error?: (message: string, title?: string) => void\n warning?: (message: string, title?: string) => void\n info?: (message: string, title?: string) => void\n }\n\n interface HomebridgePluginUiAPI {\n request: (endpoint: string, params?: any) => Promise<any>\n getPluginConfig?: () => Promise<any[]>\n updatePluginConfig?: (config: any[]) => Promise<void>\n savePluginConfig?: () => Promise<void>\n closeSettings?: () => void\n showSpinner?: () => void\n hideSpinner?: () => void\n disableSaveButton?: () => void\n enableSaveButton?: () => void\n toast?: HomebridgeToastApi\n }\n\n let homebridge: HomebridgePluginUiAPI\n}\n\nexport {}\n", "import './types.js'\nimport { uiLog } from './logger.js'\n\nfunction callUiMethod(name: keyof HomebridgePluginUiAPI, ...args: any[]): void {\n try {\n if (typeof homebridge?.[name] === 'function') {\n uiLog.info(`[callUiMethod] Invoking homebridge.${String(name)}()`)\n const fn = homebridge?.[name]\n if (typeof fn === 'function') {\n uiLog.info(`[callUiMethod] Invoking homebridge.${String(name)}()`);\n (fn as (...args: any[]) => void).apply(homebridge, args)\n } else {\n uiLog.warn(`[callUiMethod] homebridge[${String(name)}] is not a function.`)\n }\n } else {\n uiLog.warn(`[callUiMethod] homebridge[${String(name)}] is not a function.`)\n }\n } catch (e) {\n uiLog.warn(`Homebridge UI method ${String(name)} failed:`, e)\n }\n}\n\nexport function showBusyUi(): void {\n callUiMethod('disableSaveButton')\n callUiMethod('showSpinner')\n}\n\nexport function hideBusyUi(): void {\n callUiMethod('hideSpinner')\n callUiMethod('enableSaveButton')\n}\n\nexport function closeSettingsModal(): void {\n callUiMethod('closeSettings')\n}\n", "import './types.js'\nimport { uiLog } from './logger.js'\n\nfunction showToast(\n method: 'success' | 'error' | 'warning' | 'info',\n message: string,\n title = 'SwitchBot',\n): void {\n try {\n // Defensive: check for window and homebridge existence\n const hb = typeof window !== 'undefined' ? (window as any).homebridge : undefined\n const toast = hb && typeof hb.toast === 'object' ? hb.toast : undefined\n const fn = toast && typeof toast[method] === 'function' ? toast[method] : undefined\n if (fn) {\n try {\n fn(message, title)\n return\n } catch (err) {\n uiLog.warn(`Toast ${method} threw:`, err)\n }\n }\n // Fallback: log to console\n uiLog.info(`[Toast:${method}] ${title} - ${message}`)\n } catch (e) {\n uiLog.warn(`Toast ${method} outer error:`, e)\n uiLog.info(`[Toast:${method}] ${title} - ${message}`)\n }\n}\n\nexport function toastSuccess(message: string, title?: string): void {\n showToast('success', message, title)\n}\n\nexport function toastError(message: string, title?: string): void {\n showToast('error', message, title)\n}\n\nexport function toastWarning(message: string, title?: string): void {\n showToast('warning', message, title)\n}\n\nexport function toastInfo(message: string, title?: string): void {\n showToast('info', message, title)\n}\n", "export const DEVICE_TYPES = {\n 'Window Coverings': ['Blind Tilt', 'Curtain', 'Curtain3', 'Roller Shade'],\n 'Locks & Access': [\n 'Keypad',\n 'Keypad Touch',\n 'Keypad Vision',\n 'Keypad Vision Pro',\n 'Lock Vision Pro',\n 'Lock Lite',\n 'Smart Lock',\n 'Smart Lock Pro',\n 'Smart Lock Ultra',\n 'Video Doorbell',\n ],\n 'Sensors': ['Contact Sensor', 'Motion Sensor', 'Presence Sensor', 'Water Detector'],\n 'Lighting': [\n 'Candle Warmer Lamp',\n 'Ceiling Light',\n 'Ceiling Light Pro',\n 'Color Bulb',\n 'Floor Lamp',\n 'RGBIC Neon Rope Light',\n 'RGBIC Neon Wire Rope Light',\n 'RGBICWW Floor Lamp',\n 'RGBICWW Strip Light',\n 'Strip Light',\n 'Strip Light 3',\n ],\n 'Climate Control': [\n 'Air Purifier PM2.5',\n 'Air Purifier Table PM2.5',\n 'Air Purifier VOC',\n 'Air Purifier Table VOC',\n 'Battery Circulator Fan',\n 'Circulator Fan',\n 'Humidifier',\n 'Humidifier2',\n 'Meter',\n 'MeterPlus',\n 'Meter Plus',\n 'MeterPro',\n 'Meter Pro',\n 'MeterPro(CO2)',\n 'Meter Pro (CO2)',\n 'Smart Radiator Thermostat',\n 'Standing Circulator Fan',\n 'WoIOSensor',\n ],\n 'Plugs & Switches': [\n 'Garage Door Opener',\n 'Plug',\n 'Plug Mini (EU)',\n 'Plug Mini (JP)',\n 'Plug Mini (US)',\n 'Relay Switch 1',\n 'Relay Switch 1PM',\n 'Relay Switch 2PM',\n ],\n 'Robot Vacuums': [\n 'K10+',\n 'K10+ Pro',\n 'Robot Vacuum Cleaner K10+ Pro Combo',\n 'Robot Vacuum Cleaner K11+',\n 'Robot Vacuum Cleaner K20 Plus Pro',\n 'Robot Vacuum Cleaner S1',\n 'Robot Vacuum Cleaner S1 Plus',\n 'Robot Vacuum Cleaner S10',\n 'Robot Vacuum Cleaner S20',\n ],\n 'Hubs': ['AI Hub', 'Hub', 'Hub 2', 'Hub 3', 'Hub Mini', 'Hub Plus'],\n 'Cameras': [\n 'Indoor Cam',\n 'Pan/Tilt Cam',\n 'Pan/Tilt Cam 2K',\n 'Pan/Tilt Cam Plus 2K',\n 'Pan/Tilt Cam Plus 3K',\n ],\n 'IR Devices': [\n 'Air Conditioner',\n 'Air Purifier',\n 'Camera',\n 'DVD',\n 'Fan',\n 'Light',\n 'Others',\n 'Projector',\n 'Set Top Box',\n 'Speaker',\n 'Streamer',\n 'TV',\n 'Vacuum Cleaner',\n 'Water Heater',\n ],\n 'Other Devices': ['AI Art Frame', 'Bot', 'Home Climate Panel', 'Remote', 'remote with screen'],\n} as const\n\nexport const DEVICE_TYPE_NORMALIZATION_MAP: Record<string, string> = {\n // --- node-switchbot v4 normalization additions ---\n 'hub mini': 'Hub Mini',\n 'hub 3': 'Hub 3',\n 'keypad': 'Keypad',\n 'plug mini': 'Plug Mini (US)', // fallback to US if region not specified\n 'art frame': 'AI Art Frame',\n 'rgbicww': 'RGBICWW Strip Light',\n 'lock vision': 'Lock Vision Pro', // alias for new lock vision\n 'lock pro': 'Smart Lock Pro',\n 'lock lite': 'Lock Lite',\n 'circulator fan': 'Circulator Fan',\n 'smart thermostat radiator': 'Smart Radiator Thermostat',\n 'climate panel': 'Home Climate Panel',\n 'evaporative humidifier': 'Humidifier',\n // --- end node-switchbot v4 additions ---\n // Only keep the last occurrence for each key, all values canonical\n 'air purifier pm2.5': 'Air Purifier PM2.5',\n 'pan/tilt cam plus 3k': 'Pan/Tilt Cam Plus 3K',\n 'remote with screen': 'Remote with Screen',\n 'ai hub': 'AI Hub',\n 'water detector': 'Water Detector',\n 'video doorbell': 'Video Doorbell',\n 'smart radiator thermostat': 'Smart Radiator Thermostat',\n 'woiosensor': 'WoIOSensor',\n 'garage door opener': 'Garage Door Opener',\n 'air purifier table pm2.5': 'Air Purifier Table PM2.5',\n 'air purifier voc': 'Air Purifier VOC',\n 'air purifier table voc': 'Air Purifier Table VOC',\n 'plug mini (eu)': 'Plug Mini (EU)',\n // Only last occurrence for each key is kept above. Removed duplicates here.\n\n // Climate control conversions\n 'humidifier2': 'Humidifier2',\n 'battery circulator fan': 'Battery Circulator Fan',\n 'standing circulator fan': 'Standing Circulator Fan',\n\n // Lock/keypad conversions\n 'smart lock': 'Smart Lock',\n 'smart lock pro': 'Smart Lock Pro',\n 'smart lock ultra': 'Smart Lock Ultra',\n 'keypad touch': 'Keypad Touch',\n 'keypad vision': 'Keypad Vision',\n 'keypad vision pro': 'Keypad Vision Pro',\n\n // Light conversions\n 'color bulb': 'Color Bulb',\n 'ceiling light': 'Ceiling Light',\n 'ceiling light pro': 'Ceiling Light Pro',\n 'candle warmer lamp': 'Candle Warmer Lamp',\n 'floor lamp': 'Floor Lamp',\n 'rgbic neon rope light': 'RGBIC Neon Rope Light',\n 'rgbic neon wire rope light': 'RGBIC Neon Wire Rope Light',\n 'rgbicww floor lamp': 'RGBICWW Floor Lamp',\n 'rgbicww strip light': 'RGBICWW Strip Light',\n 'strip light': 'Strip Light',\n 'strip light 3': 'Strip Light 3',\n\n // Vacuum conversions\n 'robot vacuum cleaner s1': 'Robot Vacuum Cleaner S1',\n 'robot vacuum cleaner s1 plus': 'Robot Vacuum Cleaner S1 Plus',\n 'robot vacuum cleaner s10': 'Robot Vacuum Cleaner S10',\n 'robot vacuum cleaner s20': 'Robot Vacuum Cleaner S20',\n 'robot vacuum cleaner k10+ pro combo': 'Robot Vacuum Cleaner K10+ Pro Combo',\n 'robot vacuum cleaner k11+': 'Robot Vacuum Cleaner K11+',\n 'robot vacuum cleaner k20 plus pro': 'Robot Vacuum Cleaner K20 Plus Pro',\n\n // Exact device type mappings (API format \u2192 canonical format)\n 'relay switch 1': 'Relay Switch 1',\n 'blind tilt': 'Blind Tilt',\n 'roller shade': 'Roller Shade',\n 'curtain3': 'Curtain3',\n 'hub 2': 'Hub 2',\n 'meterplus': 'MeterPlus',\n 'meterpro': 'MeterPro',\n 'meterpro(co2)': 'MeterPro(CO2)',\n 'walletfinder': 'WalletFinder',\n 'k10+': 'K10+',\n 'k10+ pro (wosweeperminipro)': 'K10+ Pro (wosweeperminipro)',\n\n // Handle spaced variants from config files (normalize back to canonical type)\n 'meter pro': 'Meter Pro',\n 'meter pro (co2)': 'Meter Pro (CO2)',\n 'meter plus': 'Meter Plus',\n 'relay switch 1 pm': 'Relay Switch 1PM',\n 'relay switch 2 pm': 'Relay Switch 2PM',\n 'plug mini eu': 'Plug Mini (EU)',\n 'plug mini jp': 'Plug Mini (JP)',\n 'plug mini us': 'Plug Mini (US)',\n\n // Migration mappings for invalid/legacy device types\n 'lock vision pro': 'Lock Vision Pro', // Valid alias; map to canonical\n // 'lock vision': 'Keypad Vision', // Invalid type (removed, now alias above)\n 'lock touch': 'Keypad Touch', // Invalid type\n\n // Additional normalization for new/unknown types from logs\n 'woplugus': 'Plug Mini (US)',\n // Removed duplicate keys below, only last occurrence kept\n // 'plug mini us': 'plug mini (us)', // duplicate, removed\n // 'plug us': 'plug mini (us)', // duplicate, removed\n // 'plug': 'plug', // duplicate, removed\n // 'air purifier pm2.5': 'air purifier pm2.5', // duplicate, removed\n // 'rgbic neon wire rope light': 'rgbic neon wire rope light', // duplicate, removed\n // 'candle warmer lamp': 'candle warmer lamp', // duplicate, removed\n // 'pan/tilt cam plus 3k': 'pan/tilt cam plus 3k', // duplicate, removed\n // 'remote with screen': 'remote with screen', // duplicate, removed\n // 'ai hub': 'ai hub', // duplicate, removed\n // 'lock vision pro': 'lock vision pro', // duplicate, remove this line\n // Add any other device types from logs as needed\n}\n\n/**\n * Get all valid device types as a flat set for validation\n */\nexport function getValidDeviceTypes(): Set<string> {\n const validTypes = new Set<string>()\n for (const category of Object.values(DEVICE_TYPES)) {\n for (const type of category) {\n validTypes.add(type)\n }\n }\n return validTypes\n}\n\n/**\n * Normalize and validate a device type, returning a valid type or null\n * @param deviceType The device type to validate/normalize\n * @returns Valid device type string or null if no valid mapping found\n */\nexport function normalizeDeviceType(deviceType: string | undefined | null): string | null {\n if (!deviceType || typeof deviceType !== 'string') {\n return null\n }\n\n const trimmed = deviceType.trim()\n const lowercase = trimmed.toLowerCase()\n\n // Check if already valid\n const validTypes = getValidDeviceTypes()\n if (validTypes.has(trimmed)) {\n return trimmed\n }\n\n // Check normalization map\n const normalized = DEVICE_TYPE_NORMALIZATION_MAP[lowercase]\n if (normalized && validTypes.has(normalized)) {\n return normalized\n }\n\n // No valid mapping found\n return null\n}\n\n/**\n * Check if a device type is valid\n * @param deviceType The device type to check\n * @returns true if device type is in DEVICE_TYPES\n */\nexport function isValidDeviceType(deviceType: string | undefined | null): boolean {\n if (!deviceType || typeof deviceType !== 'string') {\n return false\n }\n const validTypes = getValidDeviceTypes()\n return validTypes.has(deviceType.trim())\n}\n", "// Fetch the list of configured devices from the Homebridge UI API\nimport { isValidDeviceType, normalizeDeviceType } from '../../../device-types.js'\nimport './types.js'\nimport { uiLog } from './logger.js'\nimport { toastError } from './toast.js'\n\nexport async function fetchDevices(): Promise<any[]> {\n try {\n if (typeof homebridge.getPluginConfig !== 'function') {\n throw new TypeError('Homebridge UI API not available')\n }\n const configArr = await homebridge.getPluginConfig()\n const config = Array.isArray(configArr) && configArr.length > 0 ? configArr.find(isSwitchBotPlatformConfig) : null\n if (!config || !Array.isArray(config.devices)) {\n return []\n }\n return config.devices\n } catch (e) {\n const msg = e instanceof Error ? e.message : String(e)\n uiLog.error('Error fetching devices:', msg)\n return []\n }\n}\n\n/**\n * Validate and auto-correct device types in the config array before saving.\n * Returns an array of errors for devices that cannot be fixed.\n */\nexport function validateAndFixDeviceTypes(devices: Array<{ deviceId: string, configDeviceName: string, configDeviceType: string }>) {\n const errors: Array<{ deviceId: string, name: string, type: string }> = []\n for (const d of devices) {\n if (!isValidDeviceType(d.configDeviceType)) {\n const fixed = normalizeDeviceType(d.configDeviceType)\n if (fixed) {\n d.configDeviceType = fixed\n } else {\n errors.push({\n deviceId: d.deviceId,\n name: d.configDeviceName,\n type: d.configDeviceType,\n })\n }\n }\n }\n return errors\n}\n\n// API wrapper functions for communicating with the Homebridge UI server\n\nfunction isSwitchBotPlatformConfig(block: any): boolean {\n const platformName = String(block?.platform || block?.name || '').toLowerCase()\n return (\n platformName === 'switchbot'\n || platformName === '@switchbot/homebridge-switchbot'\n || platformName.includes('switchbot')\n )\n}\n\nexport async function syncParentPluginConfigFromDisk(autoSave = false): Promise<boolean> {\n try {\n if (\n typeof homebridge.getPluginConfig !== 'function'\n || typeof homebridge.updatePluginConfig !== 'function'\n ) {\n uiLog.warn('Parent config sync API not available')\n return false\n }\n\n // Use Homebridge UI API to fetch and update config\n const pluginConfigBlocks = await homebridge.getPluginConfig()\n if (!Array.isArray(pluginConfigBlocks) || !pluginConfigBlocks.length) {\n uiLog.warn('No plugin config blocks returned from Homebridge')\n return false\n }\n\n const index = pluginConfigBlocks.findIndex(block => isSwitchBotPlatformConfig(block))\n if (index < 0) {\n uiLog.warn('SwitchBot platform block not found in Homebridge plugin config')\n return false\n }\n\n // Validate and fix device types before saving\n const errors = validateAndFixDeviceTypes(pluginConfigBlocks[index].devices || [])\n if (errors.length > 0) {\n toastError(`Invalid device types found: ${errors.map(e => `${e.name} (${e.type})`).join(', ')}`)\n return false\n }\n // pluginConfigBlocks[index] is already up to date\n await homebridge.updatePluginConfig(pluginConfigBlocks)\n\n // Auto-save to disk if requested - prevents parent UI from overwriting with stale cache\n if (autoSave && typeof homebridge.savePluginConfig === 'function') {\n uiLog.info('Auto-saving config to disk...')\n await homebridge.savePluginConfig()\n uiLog.info('Config saved successfully')\n }\n\n return true\n } catch (e) {\n uiLog.warn('Failed to sync parent plugin config cache:', e)\n return false\n }\n}\n\nexport async function fetchCredentialStatus(): Promise<any> {\n try {\n const resp = await homebridge.request('/credentials', {})\n uiLog.info('Load credentials response:', resp)\n\n if (!resp || resp.success === false) {\n uiLog.error('Failed to load credentials:', resp)\n return null\n }\n\n return resp.data || {}\n } catch (e) {\n uiLog.error('Error loading credentials:', e)\n return null\n }\n}\n\nexport async function saveCredentials(token: string, secret: string): Promise<any> {\n uiLog.info('Saving credentials...')\n const resp = await homebridge.request('/credentials', { token, secret })\n uiLog.info('Save response:', resp)\n if (!resp || resp.success === false) {\n throw new Error(resp?.message || 'Save failed')\n }\n return resp.data || resp\n}\n\nexport interface DiscoverRequestOptions {\n bleEnabled?: boolean\n bleScanDurationSeconds?: number\n bleTimeoutSeconds?: number\n}\n\nexport async function discoverDevices(\n mode: 'all' | 'ble' | 'openapi' = 'all',\n options?: DiscoverRequestOptions,\n): Promise<any[]> {\n const resp = await homebridge.request('/discover', { mode, ...options })\n uiLog.info('Discover response:', resp)\n if (!resp || resp.success === false) {\n throw new Error(resp?.data?.message || 'Discovery failed')\n }\n return resp.data || []\n}\n\nexport async function fetchBluetoothStatus(): Promise<{ available: boolean, message: string }> {\n try {\n const resp = await homebridge.request('/ble-status', {})\n if (!resp || resp.success === false) {\n return { available: false, message: 'Bluetooth status unavailable' }\n }\n return resp.data || { available: false, message: 'Bluetooth status unavailable' }\n } catch (_e) {\n return { available: false, message: 'Bluetooth status unavailable' }\n }\n}\n\nexport async function testDeviceConnection(payload: {\n deviceId: string\n connectionType?: string\n address?: string\n}): Promise<{\n success: boolean\n deviceId: string\n method: string\n latencyMs: number\n message: string\n state?: Record<string, any>\n}> {\n const resp = await homebridge.request('/test-connection', payload)\n if (!resp || resp.success === false) {\n throw new Error(resp?.data?.message || 'Connection test failed')\n }\n\n return resp.data || {\n success: false,\n deviceId: payload.deviceId,\n method: 'Auto',\n latencyMs: 0,\n message: 'Connection test failed',\n }\n}\n\nexport async function addDevice(\n deviceId: string,\n name: string,\n type: string,\n options?: { address?: string, model?: string, rssi?: number, encryptionKey?: string, keyId?: string },\n): Promise<any> {\n if (typeof homebridge.getPluginConfig !== 'function' || typeof homebridge.updatePluginConfig !== 'function') {\n throw new TypeError('Homebridge UI API not available')\n }\n const configArr = await homebridge.getPluginConfig()\n const idx = Array.isArray(configArr) ? configArr.findIndex(isSwitchBotPlatformConfig) : -1\n if (idx === -1) {\n throw new Error('SwitchBot config not found')\n }\n const config = configArr[idx]\n if (!Array.isArray(config.devices)) {\n config.devices = []\n }\n const normalizedDeviceId = String(deviceId).trim().toLowerCase()\n const exists = config.devices.some((d: any) => String(d.deviceId ?? d.id ?? '').trim().toLowerCase() === normalizedDeviceId)\n if (exists) {\n return { alreadyExists: true, message: 'Device already in config' }\n }\n const newDevice: any = { deviceId, configDeviceName: name, configDeviceType: type }\n if (options?.address) {\n newDevice.address = options.address\n }\n if (options?.model) {\n newDevice.model = options.model\n }\n if (options?.rssi !== undefined && options?.rssi !== null && options?.rssi !== 0) {\n newDevice.rssi = options.rssi\n }\n if (options?.encryptionKey) {\n newDevice.encryptionKey = options.encryptionKey\n }\n if (options?.keyId) {\n newDevice.keyId = options.keyId\n }\n config.devices.push(newDevice)\n await homebridge.updatePluginConfig(configArr)\n if (typeof homebridge.savePluginConfig === 'function') {\n await homebridge.savePluginConfig()\n }\n return { added: true, message: `Device \"${name}\" added successfully` }\n}\n\nexport async function addDevicesInBulk(\n devices: Array<{ deviceId: string, name: string, type: string, rssi?: number, address?: string, model?: string }>,\n): Promise<any> {\n const resp = await homebridge.request('/add-devices', { devices })\n uiLog.info('Bulk add response:', resp)\n if (!resp || resp.success === false) {\n throw new Error(resp?.data?.message || 'Bulk add failed')\n }\n return resp.data || resp\n}\n\nexport async function updateDevice(\n deviceId: string,\n configDeviceName?: string,\n configDeviceType?: string,\n options?: {\n refreshRate?: number\n connectionPreference?: string\n encryptionKey?: string\n keyId?: string\n room?: string\n [key: string]: any\n },\n): Promise<any> {\n if (typeof homebridge.getPluginConfig !== 'function' || typeof homebridge.updatePluginConfig !== 'function') {\n throw new TypeError('Homebridge UI API not available')\n }\n const configArr = await homebridge.getPluginConfig()\n const idx = Array.isArray(configArr) ? configArr.findIndex(isSwitchBotPlatformConfig) : -1\n if (idx === -1) {\n throw new Error('SwitchBot config not found')\n }\n const config = configArr[idx]\n if (!Array.isArray(config.devices)) {\n throw new TypeError('No devices array in config')\n }\n const normalizedDeviceId = String(deviceId).trim().toLowerCase()\n const device = config.devices.find((d: any) => String(d.deviceId ?? d.id ?? '').trim().toLowerCase() === normalizedDeviceId)\n if (!device) {\n throw new Error('Device not found in config')\n }\n if (configDeviceName) {\n device.configDeviceName = configDeviceName\n }\n if (configDeviceType) {\n device.configDeviceType = configDeviceType\n }\n if (options) {\n Object.assign(device, options)\n }\n await homebridge.updatePluginConfig(configArr)\n if (typeof homebridge.savePluginConfig === 'function') {\n await homebridge.savePluginConfig()\n }\n return { updated: true, message: `Device updated successfully` }\n}\n\nexport async function deleteDevice(deviceId: string): Promise<any> {\n if (typeof homebridge.getPluginConfig !== 'function' || typeof homebridge.updatePluginConfig !== 'function') {\n throw new TypeError('Homebridge UI API not available')\n }\n const configArr = await homebridge.getPluginConfig()\n const idx = Array.isArray(configArr) ? configArr.findIndex(isSwitchBotPlatformConfig) : -1\n if (idx === -1) {\n throw new Error('SwitchBot config not found')\n }\n const config = configArr[idx]\n if (!Array.isArray(config.devices)) {\n throw new TypeError('No devices array in config')\n }\n const normalizedDeviceId = String(deviceId).trim().toLowerCase()\n const before = config.devices.length\n // Remove the target device\n config.devices = config.devices.filter(d => String(d.deviceId ?? d.id ?? '').trim().toLowerCase() !== normalizedDeviceId)\n // Defensive: filter out any invalid device entries (missing required fields)\n config.devices = config.devices.filter(d => d && typeof d === 'object' && d.deviceId && d.configDeviceType)\n if (config.devices.length === before) {\n throw new Error('Device not found in config')\n }\n await homebridge.updatePluginConfig(configArr)\n if (typeof homebridge.savePluginConfig === 'function') {\n await homebridge.savePluginConfig()\n }\n return { deleted: true, message: `Device removed from config` }\n}\n\nexport async function deleteAllDevices(): Promise<any> {\n if (typeof homebridge.getPluginConfig !== 'function' || typeof homebridge.updatePluginConfig !== 'function') {\n throw new TypeError('Homebridge UI API not available')\n }\n const configArr = await homebridge.getPluginConfig()\n // Find or create the SwitchBot config block\n let idx = Array.isArray(configArr) ? configArr.findIndex(isSwitchBotPlatformConfig) : -1\n if (idx === -1) {\n // If not found, create a new config block for SwitchBot\n const newBlock = { platform: 'SwitchBot', devices: [] }\n configArr.push(newBlock)\n idx = configArr.length - 1\n }\n const config = configArr[idx]\n // Always ensure devices is an array\n if (!Array.isArray(config.devices)) {\n config.devices = []\n }\n const deletedCount = config.devices.length\n config.devices = []\n // Defensive: ensure required fields for schema compliance\n if (!config.platform) {\n config.platform = 'SwitchBot'\n }\n // Ensure 'name' property is present (required by schema for Homebridge platform blocks)\n if (!config.name) {\n config.name = 'SwitchBot'\n }\n // Save updated config\n await homebridge.updatePluginConfig(configArr)\n if (typeof homebridge.savePluginConfig === 'function') {\n await homebridge.savePluginConfig()\n }\n return { deleted: true, deletedCount, message: `Removed ${deletedCount} device(s) from config` }\n}\n", "// Extend the Window interface to include _discoverySelectedIds for type safety\n// Batch enable/disable helper (true module scope for UI access)\nimport {\n addDevicesInBulk,\n discoverDevices as apiDiscoverDevices,\n fetchBluetoothStatus,\n // fetchDevices, // removed unused import\n syncParentPluginConfigFromDisk,\n} from './api.js'\nimport { loadConfiguredDevices } from './devices.js'\nimport { uiLog } from './logger.js'\nimport { hideBusyUi, showBusyUi } from './modal.js'\nimport { getDiscoveryPreferences, renderDiscoveredDevices, setDiscoveryPreferences } from './render.js'\nimport { toastError, toastInfo, toastSuccess, toastWarning } from './toast.js'\n\ndeclare global {\n interface Window {\n _discoverySelectedIds: Set<string>\n }\n}\n\nfunction normalizeId(value: any): string {\n return String(value ?? '').trim().toLowerCase()\n}\n\nfunction dedupeById(devices: any[]): any[] {\n return devices.filter((d, index, arr) => !!d?.id && arr.findIndex(x => x?.id === d.id) === index)\n}\n\nfunction mergeDiscoveredDevices(existingDevices: any[], incomingDevices: any[]): any[] {\n const deviceMap = new Map<string, any>()\n\n for (const d of dedupeById(existingDevices)) {\n deviceMap.set(d.id, { ...d })\n }\n\n for (const d of dedupeById(incomingDevices)) {\n const current = deviceMap.get(d.id)\n if (current) {\n // Only set 'Both' if both sources are present in this session\n let nextConnectionType = current.connectionType\n if (current.connectionType && d.connectionType && current.connectionType !== d.connectionType) {\n // Only set 'Both' if both are 'BLE' and 'OpenAPI' (not if one is undefined)\n const types = [current.connectionType, d.connectionType].sort().join(',')\n if (types === 'BLE,OpenAPI' || types === 'OpenAPI,BLE') {\n nextConnectionType = 'Both'\n }\n }\n deviceMap.set(d.id, {\n ...current,\n ...d,\n connectionType: nextConnectionType,\n })\n } else {\n deviceMap.set(d.id, { ...d })\n }\n }\n\n const merged = [...deviceMap.values()]\n // Log merged device structure for debugging\n if (merged.length > 0) {\n console.warn('[SwitchBot][Discovery][mergeDiscoveredDevices] Merged device sample:', merged[0])\n\n console.warn('[SwitchBot][Discovery][mergeDiscoveredDevices] Total merged devices:', merged.length)\n }\n return merged\n}\n\ntype DiscoveryGroupBy = 'connection' | 'hub' | 'type'\n\ninterface DiscoveryBleSettings {\n bleEnabled: boolean\n bleScanDurationSeconds: number\n bleTimeoutSeconds: number\n}\n\nconst DISCOVERY_GROUP_BY_KEY = 'discoveryGroupBy'\nconst DISCOVERY_GROUP_EXPANDED_KEY = 'discoveryGroupExpanded'\nconst DISCOVERY_BLE_SETTINGS_KEY = 'discoveryBleSettings'\nconst DISCOVERY_HIDE_ADDED_KEY = 'discoveryHideAdded'\nconst DISCOVERY_CACHE_KEY = 'discoveryCache'\nconst DISCOVERY_AUTO_REFRESH_KEY = 'discoveryAutoRefreshSeconds'\nconst DISCOVERY_CACHE_TTL_MS = 5 * 60 * 1000\n\nlet discoveryAutoRefreshTimer: ReturnType<typeof setInterval> | null = null\nlet discoveryLastScannedTimer: ReturnType<typeof setInterval> | null = null\n\ninterface DiscoveryCachePayload {\n timestamp: number\n devices: any[]\n}\n\nfunction setDiscoveryCache(devices: any[]): void {\n try {\n const payload: DiscoveryCachePayload = { timestamp: Date.now(), devices }\n localStorage.setItem(DISCOVERY_CACHE_KEY, JSON.stringify(payload))\n } catch (_e) {\n // Ignore storage errors\n }\n}\n\nfunction clearDiscoveryCache(): void {\n try {\n localStorage.removeItem(DISCOVERY_CACHE_KEY)\n } catch (_e) {\n // Ignore storage errors\n }\n}\n\nfunction getDiscoveryCache(validOnly = true): DiscoveryCachePayload | null {\n try {\n const stored = localStorage.getItem(DISCOVERY_CACHE_KEY)\n if (!stored) {\n return null\n }\n const payload = JSON.parse(stored) as DiscoveryCachePayload\n if (!payload || !Array.isArray(payload.devices) || typeof payload.timestamp !== 'number') {\n return null\n }\n const age = Date.now() - payload.timestamp\n if (validOnly && age > DISCOVERY_CACHE_TTL_MS) {\n return null\n }\n return payload\n } catch (_e) {\n return null\n }\n}\n\nfunction getDiscoveryAutoRefreshSeconds(): number {\n try {\n const stored = localStorage.getItem(DISCOVERY_AUTO_REFRESH_KEY)\n const value = Number(stored || 0)\n return Number.isFinite(value) && value >= 0 ? value : 0\n } catch (_e) {\n return 0\n }\n}\n\nfunction setDiscoveryAutoRefreshSeconds(value: number): void {\n try {\n localStorage.setItem(DISCOVERY_AUTO_REFRESH_KEY, String(Math.max(0, value)))\n } catch (_e) {\n // Ignore storage errors\n }\n}\n\nfunction getDiscoveryHideAddedPreference(): boolean {\n try {\n return localStorage.getItem(DISCOVERY_HIDE_ADDED_KEY) === 'true'\n } catch (_e) {\n return false\n }\n}\n\nfunction setDiscoveryHideAddedPreference(value: boolean): void {\n try {\n localStorage.setItem(DISCOVERY_HIDE_ADDED_KEY, String(value))\n } catch (_e) {\n // Ignore storage errors\n }\n}\n\nfunction formatElapsedShort(ms: number): string {\n const totalSeconds = Math.max(0, Math.floor(ms / 1000))\n if (totalSeconds < 60) {\n return `${totalSeconds}s ago`\n }\n const minutes = Math.floor(totalSeconds / 60)\n if (minutes < 60) {\n return `${minutes}m ago`\n }\n const hours = Math.floor(minutes / 60)\n if (hours < 24) {\n return `${hours}h ago`\n }\n const days = Math.floor(hours / 24)\n return `${days}d ago`\n}\n\nfunction updateLastScannedStatus(): void {\n const lastScannedStatus = document.getElementById('lastScannedStatus')\n if (!lastScannedStatus) {\n return\n }\n\n const cache = getDiscoveryCache(false)\n if (!cache) {\n lastScannedStatus.textContent = 'Last scanned: never'\n return\n }\n\n const ageMs = Date.now() - cache.timestamp\n const timestampText = new Date(cache.timestamp).toLocaleString()\n const stale = ageMs > DISCOVERY_CACHE_TTL_MS\n lastScannedStatus.textContent = stale\n ? `Last scanned: ${formatElapsedShort(ageMs)} (${timestampText}, cache expired)`\n : `Last scanned: ${formatElapsedShort(ageMs)} (${timestampText})`\n}\n\nasync function renderCachedDiscoveryResults(): Promise<void> {\n const cache = getDiscoveryCache(true)\n const list = document.getElementById('discoveredList')\n if (!cache || !list || !cache.devices.length) {\n return\n }\n\n list.style.display = 'block'\n // Use type-safe property for window._discoverySelectedIds\n if (!(window as any)._discoverySelectedIds) {\n (window as any)._discoverySelectedIds = new Set()\n }\n await updateDiscoveryView(\n cache.devices,\n getDiscoveryPreferences(),\n getDiscoveryGroupByPreference(),\n getDiscoveryHideAddedPreference(),\n (window as any)._discoverySelectedIds,\n )\n}\n\nfunction getDiscoveryBleSettings(): DiscoveryBleSettings {\n try {\n const stored = localStorage.getItem(DISCOVERY_BLE_SETTINGS_KEY)\n if (!stored) {\n return { bleEnabled: true, bleScanDurationSeconds: 5, bleTimeoutSeconds: 8 }\n }\n const parsed = JSON.parse(stored)\n return {\n bleEnabled: parsed?.bleEnabled !== false,\n bleScanDurationSeconds: Math.max(3, Math.min(15, Number(parsed?.bleScanDurationSeconds || 5))),\n bleTimeoutSeconds: Math.max(3, Math.min(30, Number(parsed?.bleTimeoutSeconds || 8))),\n }\n } catch (_e) {\n return { bleEnabled: true, bleScanDurationSeconds: 5, bleTimeoutSeconds: 8 }\n }\n}\n\nfunction setDiscoveryBleSettings(settings: DiscoveryBleSettings): void {\n try {\n localStorage.setItem(DISCOVERY_BLE_SETTINGS_KEY, JSON.stringify(settings))\n } catch (_e) {\n // Ignore storage errors\n }\n}\n\nexport async function initializeDiscoverySettings(): Promise<void> {\n const scanSelect = document.getElementById('bleScanDurationSelect') as HTMLSelectElement | null\n const timeoutInput = document.getElementById('bleTimeoutInput') as HTMLInputElement | null\n const disableBleCheckbox = document.getElementById('disableBleScanCheckbox') as HTMLInputElement | null\n const scanSetting = document.getElementById('bleScanSetting')\n const timeoutSetting = document.getElementById('bleTimeoutSetting')\n const bluetoothStatus = document.getElementById('bluetoothStatus')\n const autoRefreshSelect = document.getElementById('autoRefreshIntervalSelect') as HTMLSelectElement | null\n const refreshBtn = document.getElementById('refreshDiscoverBtn') as HTMLButtonElement | null\n\n const current = getDiscoveryBleSettings()\n if (scanSelect) {\n scanSelect.value = String(current.bleScanDurationSeconds)\n }\n if (timeoutInput) {\n timeoutInput.value = String(current.bleTimeoutSeconds)\n }\n if (disableBleCheckbox) {\n disableBleCheckbox.checked = !current.bleEnabled\n }\n\n if (autoRefreshSelect) {\n autoRefreshSelect.value = String(getDiscoveryAutoRefreshSeconds())\n }\n\n const updateBleSettingVisibility = (): void => {\n const disabled = !!disableBleCheckbox?.checked\n if (scanSetting) {\n scanSetting.style.display = disabled ? 'none' : 'inline-flex'\n }\n if (timeoutSetting) {\n timeoutSetting.style.display = disabled ? 'none' : 'inline-flex'\n }\n }\n\n const persistFromControls = (): void => {\n const next: DiscoveryBleSettings = {\n bleEnabled: !(disableBleCheckbox?.checked ?? false),\n bleScanDurationSeconds: Math.max(3, Math.min(15, Number(scanSelect?.value || 5))),\n bleTimeoutSeconds: Math.max(3, Math.min(30, Number(timeoutInput?.value || 8))),\n }\n setDiscoveryBleSettings(next)\n }\n\n scanSelect?.addEventListener('change', persistFromControls)\n timeoutInput?.addEventListener('change', persistFromControls)\n disableBleCheckbox?.addEventListener('change', () => {\n persistFromControls()\n updateBleSettingVisibility()\n })\n\n updateBleSettingVisibility()\n\n if (bluetoothStatus) {\n const status = await fetchBluetoothStatus()\n bluetoothStatus.textContent = status.available\n ? `Bluetooth: available (${status.message})`\n : `Bluetooth: unavailable (${status.message})`\n }\n\n updateLastScannedStatus()\n if (discoveryLastScannedTimer) {\n clearInterval(discoveryLastScannedTimer)\n }\n discoveryLastScannedTimer = setInterval(updateLastScannedStatus, 15000)\n\n refreshBtn?.addEventListener('click', () => {\n void discoverDevices()\n })\n\n const applyAutoRefresh = (): void => {\n const seconds = Math.max(0, Number(autoRefreshSelect?.value || 0))\n setDiscoveryAutoRefreshSeconds(seconds)\n\n if (discoveryAutoRefreshTimer) {\n clearInterval(discoveryAutoRefreshTimer)\n discoveryAutoRefreshTimer = null\n }\n\n if (seconds > 0) {\n discoveryAutoRefreshTimer = setInterval(() => {\n const discoverBtn = document.getElementById('discoverBtn') as HTMLButtonElement | null\n if (!discoverBtn || discoverBtn.disabled || document.hidden) {\n return\n }\n void discoverDevices()\n }, seconds * 1000)\n }\n }\n\n autoRefreshSelect?.addEventListener('change', applyAutoRefresh)\n applyAutoRefresh()\n\n await renderCachedDiscoveryResults()\n}\n\nfunction getDiscoveryGroupByPreference(): DiscoveryGroupBy {\n try {\n const stored = localStorage.getItem(DISCOVERY_GROUP_BY_KEY)\n if (stored === 'hub' || stored === 'type') {\n return stored\n }\n return 'type' // Default to Device Type grouping\n } catch (_e) {\n return 'type'\n }\n}\n\nfunction setDiscoveryGroupByPreference(groupBy: DiscoveryGroupBy): void {\n try {\n localStorage.setItem(DISCOVERY_GROUP_BY_KEY, groupBy)\n } catch (_e) {\n // Ignore storage errors\n }\n}\n\nfunction getDiscoveryGroupExpandedState(): Record<string, boolean> {\n try {\n const stored = localStorage.getItem(DISCOVERY_GROUP_EXPANDED_KEY)\n if (!stored) {\n return {}\n }\n const parsed = JSON.parse(stored)\n return typeof parsed === 'object' && parsed ? parsed : {}\n } catch (_e) {\n return {}\n }\n}\n\nfunction setDiscoveryGroupExpandedState(state: Record<string, boolean>): void {\n try {\n localStorage.setItem(DISCOVERY_GROUP_EXPANDED_KEY, JSON.stringify(state))\n } catch (_e) {\n // Ignore storage errors\n }\n}\n\nfunction isDiscoveryGroupExpanded(groupKey: string): boolean {\n const state = getDiscoveryGroupExpandedState()\n return state[groupKey] !== false\n}\n\nfunction setDiscoveryGroupExpanded(groupKey: string, expanded: boolean): void {\n const state = getDiscoveryGroupExpandedState()\n state[groupKey] = expanded\n setDiscoveryGroupExpandedState(state)\n}\n\nexport async function discoverDevices(): Promise<void> {\n const btn = document.getElementById('discoverBtn') as HTMLButtonElement\n const cancelBtn = document.getElementById('cancelDiscoverBtn') as HTMLButtonElement | null\n const status = document.getElementById('discoverStatus')\n const phaseProgress = document.getElementById('discoverPhaseProgress') as HTMLElement | null\n const phaseFill = document.getElementById('discoverPhaseFill') as HTMLElement | null\n const phaseLabel = document.getElementById('discoverPhaseLabel') as HTMLElement | null\n const list = document.getElementById('discoveredList')\n const autoAddAll = (document.getElementById('autoAddAllCheckbox') as HTMLInputElement)?.checked\n const scanSelect = document.getElementById('bleScanDurationSelect') as HTMLSelectElement | null\n const timeoutInput = document.getElementById('bleTimeoutInput') as HTMLInputElement | null\n const disableBleCheckbox = document.getElementById('disableBleScanCheckbox') as HTMLInputElement | null\n\n if (!btn) {\n console.error('[SwitchBot][Discovery] discoverDevices: discoverBtn not found in DOM')\n return\n }\n if (!status) {\n console.error('[SwitchBot][Discovery] discoverDevices: discoverStatus not found in DOM')\n return\n }\n if (!list) {\n console.error('[SwitchBot][Discovery] discoverDevices: discoveredList container not found in DOM')\n toastError('Discovery UI error: device list container missing. Please reload the page.')\n return\n }\n\n const spinnerFrames = ['\u280B', '\u2819', '\u2839', '\u2838', '\u283C', '\u2834', '\u2826', '\u2827', '\u2807', '\u280F']\n let spinnerIndex = 0\n const startedAt = Date.now()\n let phaseStartedAt = startedAt\n let phase = 'Preparing discovery...'\n let cancelled = false\n\n // --- Real-time RSSI polling additions ---\n // (bleScanDurationSeconds is now only used in bleSettings below)\n\n const setPhase = (nextPhase: string): void => {\n phase = nextPhase\n phaseStartedAt = Date.now()\n }\n\n const getPhasePercent = (phaseName: string): number => {\n if (phaseName.includes('Scanning BLE')) {\n return 35\n }\n if (phaseName.includes('Fetching OpenAPI')) {\n return 75\n }\n if (phaseName.includes('Complete')) {\n return 100\n }\n return 10\n }\n\n const renderProgress = (): void => {\n const totalSeconds = Math.max(0, Math.floor((Date.now() - startedAt) / 1000))\n const phaseSeconds = Math.max(0, Math.floor((Date.now() - phaseStartedAt) / 1000))\n const frame = spinnerFrames[spinnerIndex % spinnerFrames.length]\n spinnerIndex += 1\n status.textContent = `${frame} ${phase} (${phaseSeconds}s, ${totalSeconds}s total)`\n\n if (phaseProgress) {\n phaseProgress.style.display = 'block'\n }\n if (phaseFill) {\n phaseFill.style.width = `${getPhasePercent(phase)}%`\n }\n if (phaseLabel) {\n phaseLabel.textContent = phase\n }\n }\n\n const progressTimer = setInterval(renderProgress, 250)\n let discoveredDevices: any[] = []\n const preferences = getDiscoveryPreferences()\n let groupBy: DiscoveryGroupBy = getDiscoveryGroupByPreference()\n let hideAdded = getDiscoveryHideAddedPreference()\n // Use persistent selection state across renders\n if (!window._discoverySelectedIds) {\n window._discoverySelectedIds = new Set<string>()\n }\n const selectedIds: Set<string> = window._discoverySelectedIds\n let controlsInitialized = false\n\n // --- Real-time RSSI polling loop ---\n // (Moved inside main try block after bleSettings is defined)\n\n // Batch enable/disable helper (moved to module scope for UI access)\n async function batchSetDeviceEnabled(selectedIds: Set<string>, enabled: boolean): Promise<void> {\n // Fetch current config\n // Fetch current config using Homebridge UI API\n if (typeof homebridge.getPluginConfig !== 'function') {\n throw new TypeError('homebridge.getPluginConfig is not available')\n }\n const configArr = await homebridge.getPluginConfig()\n const platformIdx = Array.isArray(configArr) ? configArr.findIndex(c => (c.platform || c.name || '').toLowerCase().includes('switchbot')) : -1\n if (platformIdx === -1) {\n throw new Error('SwitchBot platform config not found')\n }\n const platformConfig = configArr[platformIdx]\n if (!Array.isArray(platformConfig.devices)) {\n throw new TypeError('No devices array in config')\n }\n let changed = false\n for (const dev of platformConfig.devices) {\n const id = String(dev.deviceId || dev.id || '').trim().toLowerCase()\n if (selectedIds.has(id)) {\n if (dev.enabled !== enabled) {\n dev.enabled = enabled\n changed = true\n }\n }\n }\n if (changed) {\n if (typeof homebridge.updatePluginConfig === 'function') {\n await homebridge.updatePluginConfig(configArr)\n } else {\n throw new TypeError('homebridge.updatePluginConfig is not available')\n }\n if (typeof homebridge.savePluginConfig === 'function') {\n await homebridge.savePluginConfig()\n }\n }\n }\n\n const ensureDiscoveryControls = async (): Promise<void> => {\n // --- Select All / Deselect All controls ---\n const selectAllBtn = document.createElement('button')\n selectAllBtn.textContent = 'Select All'\n selectAllBtn.style.fontSize = '13px'\n selectAllBtn.style.padding = '6px 18px'\n selectAllBtn.style.borderRadius = '6px'\n selectAllBtn.style.background = '#f3f4f6'\n selectAllBtn.style.color = '#1d4ed8'\n selectAllBtn.style.border = '1px solid #d1d5db'\n selectAllBtn.style.cursor = 'pointer'\n selectAllBtn.style.marginRight = '8px'\n selectAllBtn.onclick = () => {\n // Add all visible device IDs to selectedIds\n for (const d of discoveredDevices) {\n selectedIds.add(normalizeId(d.id))\n }\n window.dispatchEvent(new Event('discovery-selection-changed'))\n void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds)\n }\n\n const deselectAllBtn = document.createElement('button')\n deselectAllBtn.textContent = 'Deselect All'\n deselectAllBtn.style.fontSize = '13px'\n deselectAllBtn.style.padding = '6px 18px'\n deselectAllBtn.style.borderRadius = '6px'\n deselectAllBtn.style.background = '#f3f4f6'\n deselectAllBtn.style.color = '#ef4444'\n deselectAllBtn.style.border = '1px solid #d1d5db'\n deselectAllBtn.style.cursor = 'pointer'\n deselectAllBtn.onclick = () => {\n // Remove all visible device IDs from selectedIds\n for (const d of discoveredDevices) {\n selectedIds.delete(normalizeId(d.id))\n }\n window.dispatchEvent(new Event('discovery-selection-changed'))\n void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds)\n }\n\n // Insert select/deselect all controls above the action buttons\n const selectControlsRow = document.createElement('div')\n selectControlsRow.style.display = 'flex'\n selectControlsRow.style.gap = '10px'\n selectControlsRow.style.margin = '0 0 10px 0'\n selectControlsRow.appendChild(selectAllBtn)\n selectControlsRow.appendChild(deselectAllBtn)\n if (controlsInitialized) {\n return\n }\n // Always use persistent selectedIds (already defined in outer scope)\n\n const controlsDiv = document.createElement('div')\n controlsDiv.style.cssText = 'margin-bottom: 12px; display: flex; gap: 12px; flex-wrap: wrap; align-items: center;'\n\n const filterLabel = document.createElement('label')\n filterLabel.style.fontSize = '12px'\n filterLabel.style.fontWeight = '500'\n filterLabel.textContent = 'Filter:'\n\n const filterGroup = document.createElement('div')\n filterGroup.style.display = 'flex'\n filterGroup.style.gap = '4px'\n\n const filterOptions: Array<{ label: string, value: 'all' | 'ble' | 'api' | 'both' | 'ir' }> = [\n { label: 'All', value: 'all' },\n { label: 'BLE', value: 'ble' },\n { label: 'API', value: 'api' },\n { label: 'Both', value: 'both' },\n { label: 'IR', value: 'ir' },\n ]\n\n for (const option of filterOptions) {\n const filterBtn = document.createElement('button')\n filterBtn.textContent = option.label\n filterBtn.style.padding = '4px 8px'\n filterBtn.style.fontSize = '11px'\n filterBtn.style.borderRadius = '3px'\n filterBtn.style.cursor = 'pointer'\n filterBtn.style.border = preferences.connectionType === option.value ? '2px solid #007AFF' : '1px solid #ccc'\n filterBtn.style.backgroundColor = preferences.connectionType === option.value ? '#f0f7ff' : '#fff'\n filterBtn.style.color = preferences.connectionType === option.value ? '#1d4ed8' : '#374151'\n\n filterBtn.onclick = () => {\n preferences.connectionType = option.value\n setDiscoveryPreferences(preferences)\n void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds)\n\n Array.prototype.forEach.call(filterGroup.querySelectorAll('button'), (b) => {\n (b as HTMLButtonElement).style.border = '1px solid #ccc';\n (b as HTMLButtonElement).style.backgroundColor = '#fff';\n (b as HTMLButtonElement).style.color = '#374151'\n })\n filterBtn.style.border = '2px solid #007AFF'\n filterBtn.style.backgroundColor = '#f0f7ff'\n filterBtn.style.color = '#1d4ed8'\n }\n\n filterGroup.appendChild(filterBtn)\n }\n\n const sortLabel = document.createElement('label')\n sortLabel.style.fontSize = '12px'\n sortLabel.style.fontWeight = '500'\n sortLabel.style.marginLeft = '8px'\n sortLabel.textContent = 'Sort:'\n\n const sortSelect = document.createElement('select')\n sortSelect.style.fontSize = '11px'\n sortSelect.style.padding = '4px 8px'\n sortSelect.style.borderRadius = '3px'\n sortSelect.value = preferences.sortBy\n\n const sortOptions = [\n { label: 'Name', value: 'name' },\n { label: 'Signal Strength', value: 'signal' },\n { label: 'Type', value: 'type' },\n { label: 'Connection', value: 'connection' },\n ]\n\n for (const opt of sortOptions) {\n const sortOption = document.createElement('option')\n sortOption.value = opt.value\n sortOption.textContent = opt.label\n sortSelect.appendChild(sortOption)\n }\n\n sortSelect.onchange = () => {\n preferences.sortBy = sortSelect.value as any\n setDiscoveryPreferences(preferences)\n void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds)\n }\n\n const groupSelect = document.createElement('select')\n groupSelect.style.fontSize = '11px'\n groupSelect.style.padding = '4px 8px'\n groupSelect.style.borderRadius = '3px'\n // Set default value to 'type' if no stored preference\n if (!localStorage.getItem(DISCOVERY_GROUP_BY_KEY)) {\n groupSelect.value = 'type'\n } else {\n groupSelect.value = groupBy\n }\n\n const groupLabel = document.createElement('label')\n groupLabel.style.fontSize = '12px'\n groupLabel.style.fontWeight = '500'\n groupLabel.style.marginLeft = '8px'\n // Set label text to match selected group\n const groupLabelTextMap = {\n connection: 'Connection',\n hub: 'Hub',\n type: 'Device Type',\n }\n groupLabel.textContent = `Group: ${groupLabelTextMap[groupSelect.value] || 'Connection'}`\n\n const groupOptions: Array<{ label: string, value: DiscoveryGroupBy }> = [\n { label: 'Connection', value: 'connection' },\n { label: 'Hub', value: 'hub' },\n { label: 'Device Type', value: 'type' },\n ]\n\n for (const opt of groupOptions) {\n const groupOption = document.createElement('option')\n groupOption.value = opt.value\n groupOption.textContent = opt.label\n groupSelect.appendChild(groupOption)\n }\n\n groupSelect.onchange = () => {\n groupBy = groupSelect.value as DiscoveryGroupBy\n setDiscoveryGroupByPreference(groupBy)\n groupLabel.textContent = `Group: ${groupLabelTextMap[groupSelect.value] || 'Connection'}`\n void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds)\n }\n\n const hideAddedLabel = document.createElement('label')\n hideAddedLabel.style.display = 'inline-flex'\n hideAddedLabel.style.alignItems = 'center'\n hideAddedLabel.style.gap = '4px'\n hideAddedLabel.style.fontSize = '11px'\n hideAddedLabel.style.marginLeft = '8px'\n\n const hideAddedCheckbox = document.createElement('input')\n hideAddedCheckbox.type = 'checkbox'\n hideAddedCheckbox.checked = hideAdded\n hideAddedCheckbox.style.margin = '0'\n hideAddedCheckbox.style.width = 'auto'\n\n const hideAddedText = document.createElement('span')\n hideAddedText.textContent = 'Hide Added'\n\n hideAddedLabel.appendChild(hideAddedCheckbox)\n hideAddedLabel.appendChild(hideAddedText)\n\n hideAddedCheckbox.onchange = () => {\n hideAdded = hideAddedCheckbox.checked\n setDiscoveryHideAddedPreference(hideAdded)\n void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds)\n }\n\n const searchInput = document.createElement('input')\n searchInput.type = 'text'\n searchInput.placeholder = 'Search by name, ID, or type...'\n searchInput.style.fontSize = '13px'\n searchInput.style.padding = '8px 16px'\n searchInput.style.borderRadius = '6px'\n searchInput.style.border = '1px solid #ccc'\n searchInput.style.flex = '1 1 0%'\n searchInput.style.minWidth = '120px'\n searchInput.style.maxWidth = '100%'\n searchInput.style.width = '100%'\n searchInput.value = preferences.searchQuery\n\n let searchTimeout: NodeJS.Timeout\n searchInput.oninput = () => {\n clearTimeout(searchTimeout)\n searchTimeout = setTimeout(() => {\n preferences.searchQuery = searchInput.value\n setDiscoveryPreferences(preferences)\n void updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds)\n }, 300)\n }\n\n // Shared style for all top action buttons (smaller, more compact)\n const actionBtnStyle = {\n fontSize: '16px',\n padding: '10px 0',\n borderRadius: '10px',\n margin: '0 12px 0 0',\n width: '100%',\n maxWidth: '220px',\n fontWeight: 'bold',\n background: '#ef4444',\n color: '#fff',\n border: 'none',\n cursor: 'pointer',\n boxShadow: '0 2px 8px #0001',\n transition: 'background 0.2s',\n outline: 'none',\n display: 'block',\n }\n\n // Add Selected button\n const addSelectedBtn = document.createElement('button')\n addSelectedBtn.textContent = 'Add Selected to Config'\n Object.assign(addSelectedBtn.style, actionBtnStyle)\n addSelectedBtn.disabled = true\n addSelectedBtn.onclick = async () => {\n if (!selectedIds.size) {\n return\n }\n addSelectedBtn.disabled = true\n addSelectedBtn.textContent = 'Adding...'\n try {\n showBusyUi()\n const selectedDevices = discoveredDevices.filter(d => selectedIds.has(normalizeId(d.id)))\n const bulkResult = await addDevicesInBulk(selectedDevices.map(d => ({\n deviceId: d.id,\n name: d.name,\n type: d.type,\n rssi: d.rssi,\n address: d.address,\n model: d.model,\n })))\n uiLog.info('Batch add response:', bulkResult)\n if (!bulkResult || bulkResult.success === false) {\n throw new Error(bulkResult?.data?.message || 'Batch add failed')\n }\n const addedCount = bulkResult?.addedCount ?? bulkResult?.data?.addedCount ?? 0\n const skippedCount = bulkResult?.skippedCount ?? bulkResult?.data?.skippedCount ?? 0\n toastSuccess(`Added ${addedCount} device(s)${skippedCount > 0 ? ` (${skippedCount} skipped)` : ''}`)\n await loadConfiguredDevices()\n selectedIds.clear()\n addSelectedBtn.disabled = true\n addSelectedBtn.textContent = 'Add Selected'\n await updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds)\n } catch (e) {\n uiLog.error('Batch add error:', e)\n toastError(e instanceof Error ? e.message : 'Failed to add devices')\n addSelectedBtn.disabled = false\n addSelectedBtn.textContent = 'Add Selected'\n } finally {\n hideBusyUi()\n }\n }\n\n // Enable Selected button\n const enableSelectedBtn = document.createElement('button')\n enableSelectedBtn.textContent = 'Enable Selected'\n Object.assign(enableSelectedBtn.style, actionBtnStyle)\n enableSelectedBtn.disabled = true\n enableSelectedBtn.onclick = async () => {\n if (!selectedIds.size) {\n return\n }\n enableSelectedBtn.disabled = true\n enableSelectedBtn.textContent = 'Enabling...'\n try {\n showBusyUi()\n await batchSetDeviceEnabled(selectedIds, true)\n toastSuccess('Selected devices enabled')\n await loadConfiguredDevices()\n enableSelectedBtn.disabled = true\n enableSelectedBtn.textContent = 'Enable Selected'\n await updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds)\n } catch (e) {\n uiLog.error('Batch enable error:', e)\n toastError(e instanceof Error ? e.message : 'Failed to enable devices')\n enableSelectedBtn.disabled = false\n enableSelectedBtn.textContent = 'Enable Selected'\n } finally {\n hideBusyUi()\n }\n }\n\n // Disable Selected button\n const disableSelectedBtn = document.createElement('button')\n disableSelectedBtn.textContent = 'Disable Selected'\n Object.assign(disableSelectedBtn.style, actionBtnStyle)\n disableSelectedBtn.disabled = true\n disableSelectedBtn.onclick = async () => {\n if (!selectedIds.size) {\n return\n }\n disableSelectedBtn.disabled = true\n disableSelectedBtn.textContent = 'Disabling...'\n try {\n showBusyUi()\n await batchSetDeviceEnabled(selectedIds, false)\n toastSuccess('Selected devices disabled')\n await loadConfiguredDevices()\n disableSelectedBtn.disabled = true\n disableSelectedBtn.textContent = 'Disable Selected'\n await updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds)\n } catch (e) {\n uiLog.error('Batch disable error:', e)\n toastError(e instanceof Error ? e.message : 'Failed to disable devices')\n disableSelectedBtn.disabled = false\n disableSelectedBtn.textContent = 'Disable Selected'\n } finally {\n hideBusyUi()\n }\n }\n\n // Only add the filter/search controls\n controlsDiv.appendChild(filterLabel)\n controlsDiv.appendChild(filterGroup)\n controlsDiv.appendChild(sortLabel)\n controlsDiv.appendChild(sortSelect)\n controlsDiv.appendChild(groupLabel)\n controlsDiv.appendChild(groupSelect)\n controlsDiv.appendChild(hideAddedLabel)\n controlsDiv.appendChild(searchInput)\n\n // Top row action buttons container\n const topActionRow = document.createElement('div')\n\n topActionRow.style.display = 'flex'\n topActionRow.style.gap = '20px'\n topActionRow.style.margin = '18px 0 10px 0'\n topActionRow.style.justifyContent = 'flex-start'\n topActionRow.appendChild(addSelectedBtn)\n topActionRow.appendChild(enableSelectedBtn)\n topActionRow.appendChild(disableSelectedBtn)\n\n // Clear list and append controls in correct order\n list.innerHTML = ''\n list.appendChild(selectControlsRow)\n list.appendChild(topActionRow)\n list.appendChild(controlsDiv)\n\n let deviceListContainer = document.getElementById('discoveredDevices')\n if (!deviceListContainer) {\n deviceListContainer = document.createElement('ul')\n deviceListContainer.id = 'discoveredDevices'\n deviceListContainer.style.maxHeight = '400px'\n deviceListContainer.style.overflowY = 'auto'\n deviceListContainer.style.marginTop = '12px'\n deviceListContainer.style.padding = '0'\n deviceListContainer.style.listStyle = 'none'\n list.appendChild(deviceListContainer)\n }\n\n list.style.display = 'block'\n controlsInitialized = true\n\n // Update action button enabled state based on selection\n const updateActionButtons = () => {\n const hasSelection = selectedIds.size > 0\n addSelectedBtn.disabled = !hasSelection\n enableSelectedBtn.disabled = !hasSelection\n disableSelectedBtn.disabled = !hasSelection\n }\n // Listen for selection changes (selection is managed elsewhere, so poll)\n setInterval(updateActionButtons, 300)\n }\n\n try {\n const bleSettings: DiscoveryBleSettings = {\n bleEnabled: !(disableBleCheckbox?.checked ?? false),\n bleScanDurationSeconds: Math.max(3, Math.min(15, Number(scanSelect?.value || 5))),\n bleTimeoutSeconds: Math.max(3, Math.min(30, Number(timeoutInput?.value || 8))),\n }\n setDiscoveryBleSettings(bleSettings)\n\n showBusyUi()\n btn.disabled = true\n btn.textContent = '\uD83D\uDD0D Discovering...'\n setPhase(bleSettings.bleEnabled ? 'Scanning BLE...' : 'Skipping BLE scan...')\n renderProgress()\n status.classList.remove('error')\n\n const devicesFoundDisplay = document.getElementById('discoverDevicesFound')\n if (devicesFoundDisplay) {\n devicesFoundDisplay.style.display = 'none'\n devicesFoundDisplay.classList.remove('discovery-scanning-pulse')\n }\n\n if (cancelBtn) {\n cancelBtn.style.display = 'inline-block'\n cancelBtn.disabled = false\n cancelBtn.onclick = () => {\n cancelled = true\n const totalSeconds = Math.max(0, Math.floor((Date.now() - startedAt) / 1000))\n status.textContent = `Discovery cancelled (${totalSeconds}s total)`\n if (phaseProgress) {\n phaseProgress.style.display = 'none'\n }\n if (devicesFoundDisplay) {\n devicesFoundDisplay.style.display = 'none'\n devicesFoundDisplay.classList.remove('discovery-scanning-pulse')\n }\n cancelBtn.style.display = 'none'\n btn.disabled = false\n btn.textContent = '\uD83D\uDD0D Discover Devices'\n hideBusyUi()\n }\n }\n\n if (bleSettings.bleEnabled) {\n const bleDevicesRaw = await apiDiscoverDevices('ble', bleSettings)\n if (cancelled) {\n return\n }\n\n discoveredDevices = dedupeById(bleDevicesRaw)\n uiLog.info('BLE discover response:', bleDevicesRaw)\n\n // Update real-time device counter\n if (devicesFoundDisplay && bleDevicesRaw.length > 0) {\n devicesFoundDisplay.style.display = 'inline'\n devicesFoundDisplay.classList.add('discovery-scanning-pulse')\n devicesFoundDisplay.textContent = `\uD83D\uDCCA ${bleDevicesRaw.length} device(s) found (scanning...)`\n }\n } else {\n discoveredDevices = []\n uiLog.info('BLE discovery skipped by user setting')\n }\n\n if (!autoAddAll && discoveredDevices.length > 0) {\n await ensureDiscoveryControls()\n await updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds)\n status.textContent = `Showing ${discoveredDevices.length} device(s) from BLE, fetching OpenAPI...`\n }\n\n setPhase('Fetching OpenAPI...')\n renderProgress()\n\n try {\n const openApiDevicesRaw = await apiDiscoverDevices('openapi')\n if (cancelled) {\n return\n }\n\n uiLog.info('OpenAPI discover response:', openApiDevicesRaw)\n discoveredDevices = mergeDiscoveredDevices(discoveredDevices, openApiDevicesRaw)\n\n // Update device counter with merged count\n if (devicesFoundDisplay && discoveredDevices.length > 0) {\n devicesFoundDisplay.textContent = `\uD83D\uDCCA ${discoveredDevices.length} device(s) found (complete)`\n }\n } catch (openApiError) {\n uiLog.warn('OpenAPI phase failed during discovery:', openApiError)\n // Keep BLE results if available\n if (!discoveredDevices.length) {\n throw openApiError\n }\n if (devicesFoundDisplay) {\n devicesFoundDisplay.classList.remove('discovery-scanning-pulse')\n }\n }\n\n setPhase('Complete')\n renderProgress()\n uiLog.info('Final merged discover response:', discoveredDevices)\n\n if (!discoveredDevices.length) {\n status.textContent = 'No devices found in your SwitchBot account'\n toastInfo('No devices found in your SwitchBot account')\n list.style.display = 'none'\n if (devicesFoundDisplay) {\n devicesFoundDisplay.style.display = 'none'\n devicesFoundDisplay.classList.remove('discovery-scanning-pulse')\n }\n clearDiscoveryCache()\n updateLastScannedStatus()\n return\n }\n\n // If auto-add is enabled, add all devices immediately using bulk endpoint\n if (autoAddAll) {\n status.textContent = `Auto-adding ${discoveredDevices.length} device(s)...`\n\n try {\n const bulkResult = await addDevicesInBulk(\n discoveredDevices.map(d => ({\n deviceId: d.id,\n name: d.name,\n type: d.type,\n rssi: d.rssi,\n address: d.address,\n model: d.model,\n })),\n )\n uiLog.info('Bulk add response:', bulkResult)\n\n if (!bulkResult || bulkResult.success === false) {\n throw new Error(bulkResult?.data?.message || 'Bulk add failed')\n }\n\n const addedCount\n = bulkResult?.addedCount\n ?? bulkResult?.data?.addedCount\n ?? 0\n const skippedCount\n = bulkResult?.skippedCount\n ?? bulkResult?.data?.skippedCount\n ?? 0\n status.textContent = `\u2713 Added ${addedCount} device(s)${skippedCount > 0 ? ` (${skippedCount} skipped)` : ''}`\n if (addedCount > 0) {\n toastSuccess(`Added ${addedCount} device(s)${skippedCount > 0 ? ` (${skippedCount} skipped)` : ''}`)\n } else if (skippedCount > 0) {\n toastWarning(`No new devices were added (${skippedCount} skipped)`)\n }\n status.classList.remove('error')\n list.style.display = 'none'\n\n // Sync parent Homebridge form cache and auto-save to prevent cache overwrite\n // Run sync when discovery returned devices (even if backend response shape changes)\n if (discoveredDevices.length > 0) {\n const synced = await syncParentPluginConfigFromDisk(true)\n status.textContent += synced\n ? ' - Config saved automatically.'\n : ' - Warning: config may not persist until you close/reopen settings.'\n\n if (synced) {\n toastSuccess('Configuration synced and saved automatically')\n } else {\n toastWarning('Configuration sync failed; close and reopen settings before Save')\n }\n }\n } catch (e) {\n uiLog.error('Bulk add error:', e)\n status.textContent = `\u2717 Error: ${e instanceof Error ? e.message : 'Failed to add devices'}`\n status.classList.add('error')\n toastError(e instanceof Error ? e.message : 'Failed to add devices')\n }\n\n // Refresh the configured devices list\n await loadConfiguredDevices()\n return\n }\n\n await ensureDiscoveryControls()\n await updateDiscoveryView(discoveredDevices, preferences, groupBy, hideAdded, selectedIds)\n setDiscoveryCache(discoveredDevices)\n updateLastScannedStatus()\n } catch (e) {\n if (cancelled) {\n return\n }\n uiLog.error('Discovery error:', e)\n status.textContent = `Error: ${e instanceof Error ? e.message : 'Discovery failed'}`\n status.classList.add('error')\n toastError(e instanceof Error ? e.message : 'Discovery failed')\n list.style.display = 'none'\n } finally {\n clearInterval(progressTimer)\n hideBusyUi()\n if (phaseProgress) {\n phaseProgress.style.display = 'none'\n }\n if (phaseFill) {\n phaseFill.style.width = '0%'\n }\n if (phaseLabel) {\n phaseLabel.textContent = ''\n }\n const devicesFoundDisplay = document.getElementById('discoverDevicesFound')\n if (devicesFoundDisplay) {\n devicesFoundDisplay.style.display = 'none'\n devicesFoundDisplay.classList.remove('discovery-scanning-pulse')\n }\n btn.disabled = false\n btn.textContent = '\uD83D\uDD0D Discover Devices'\n if (cancelBtn) {\n cancelBtn.disabled = false\n cancelBtn.style.display = 'none'\n cancelBtn.onclick = null\n }\n }\n}\n\n/**\n * Update discovery list with filtered and sorted devices\n */\nasync function updateDiscoveryView(\n allDevices: any[],\n preferences: any,\n groupBy: any,\n hideAdded: any,\n selectedIds: Set<string>,\n) {\n // Compute visibleDevices based on filters and preferences\n console.warn('[SwitchBot][Discovery] updateDiscoveryView: allDevices', allDevices)\n const visibleDevices = allDevices\n .filter((d) => {\n // Hide already added devices if hideAdded is true\n if (hideAdded && d.added) {\n return false\n }\n // Additional filtering logic can be added here based on preferences\n return true\n })\n console.warn('[SwitchBot][Discovery] visibleDevices after filter:', visibleDevices)\n // Optionally sort devices if needed (sortDevices is imported but not used)\n // .sort((a, b) => a.name.localeCompare(b.name))\n\n // Set of already added device IDs\n const configuredIds = new Set(\n allDevices.filter(d => d.added).map(d => normalizeId(d.id)),\n )\n // Batch import controls and selection logic are handled later in the file. Removed broken/duplicated code.\n\n const getConnectionGroup = (device: any): string => {\n if (device?.isIR) {\n return 'IR'\n }\n const connectionType = String(device?.connectionType || '').toLowerCase()\n if (connectionType.includes('both')) {\n return 'Both'\n }\n if (connectionType.includes('ble')) {\n return 'BLE'\n }\n if (connectionType.includes('api')) {\n return 'OpenAPI'\n }\n return 'Unknown'\n }\n\n const getHubGroup = (device: any): string => {\n const hub = String(device?.hubDeviceId || '').trim()\n return hub ? `Hub ${hub}` : 'No Hub'\n }\n\n const getTypeGroup = (device: any): string => {\n const type = String(device?.type || '').trim()\n return type || 'Unknown Type'\n }\n\n const groupedDevices = new Map<string, any[]>()\n for (const d of visibleDevices) {\n let group = getConnectionGroup(d)\n if (groupBy === 'hub') {\n group = getHubGroup(d)\n } else if (groupBy === 'type') {\n group = getTypeGroup(d)\n }\n const groupDevices = groupedDevices.get(group) || []\n groupDevices.push(d)\n groupedDevices.set(group, groupDevices)\n }\n console.warn('[SwitchBot][Discovery] groupedDevices:', groupedDevices)\n\n let orderedGroups: string[] = []\n if (groupBy === 'hub') {\n const hubGroups = [...groupedDevices.keys()].filter(group => group !== 'No Hub').sort((a, b) => a.localeCompare(b))\n orderedGroups = groupedDevices.has('No Hub') ? [...hubGroups, 'No Hub'] : hubGroups\n } else if (groupBy === 'type') {\n const typeGroups = [...groupedDevices.keys()].filter(group => group !== 'Unknown Type').sort((a, b) => a.localeCompare(b))\n orderedGroups = groupedDevices.has('Unknown Type') ? [...typeGroups, 'Unknown Type'] : typeGroups\n } else {\n const groupOrder = ['Both', 'BLE', 'OpenAPI', 'IR', 'Unknown']\n orderedGroups = groupOrder.filter(group => groupedDevices.has(group))\n }\n\n const container = document.createElement('div')\n container.id = 'discoveredDevices'\n container.className = 'discovery-groups'\n console.warn('[SwitchBot][Discovery] Rendering device groups:', orderedGroups)\n\n if (!visibleDevices.length) {\n const empty = document.createElement('div')\n empty.className = 'discovery-group-empty'\n empty.textContent = hideAdded\n ? 'No devices match current filters (or all are already added).'\n : 'No devices match current filters.'\n container.appendChild(empty)\n console.warn('[SwitchBot][Discovery] No visible devices after filtering.')\n } else {\n for (const groupName of orderedGroups) {\n const groupItems = groupedDevices.get(groupName)\n if (!groupItems?.length) {\n continue\n }\n console.warn(`[SwitchBot][Discovery] Rendering group: ${groupName}`, groupItems)\n // ...existing code...\n const groupSection = document.createElement('section')\n groupSection.className = 'discovery-group'\n const groupStorageKey = `${groupBy}:${groupName}`\n let expanded = isDiscoveryGroupExpanded(groupStorageKey)\n const groupHeader = document.createElement('button')\n groupHeader.className = 'discovery-group-header-btn'\n groupHeader.type = 'button'\n const setGroupHeaderText = () => {\n const marker = expanded ? '\u25BE' : '\u25B8'\n groupHeader.textContent = `${marker} ${groupName} (${groupItems.length})`\n }\n setGroupHeaderText()\n groupSection.appendChild(groupHeader)\n const groupList = await renderDiscoveredDevices(groupItems, {\n configuredIds,\n selectedIds,\n onToggleSelect: (device, selected) => {\n const id = normalizeId(device.id)\n if (selected) {\n selectedIds.add(id)\n } else {\n selectedIds.delete(id)\n }\n // Update Add Selected button state\n const btn = document.querySelector('button')?.parentElement?.querySelector('button')\n if (btn && btn.textContent?.includes('Add Selected')) {\n (btn as HTMLButtonElement).disabled = selectedIds.size === 0\n }\n },\n })\n if (!expanded) {\n groupList.style.display = 'none'\n }\n groupHeader.onclick = () => {\n expanded = !expanded\n setDiscoveryGroupExpanded(groupStorageKey, expanded)\n setGroupHeaderText()\n groupList.style.display = expanded ? 'grid' : 'none'\n }\n groupSection.appendChild(groupList)\n container.appendChild(groupSection)\n }\n }\n\n // Replace or append the rendered list\n const existingList = document.getElementById('discoveredDevices')\n container.id = 'discoveredDevices'\n if (existingList && existingList.parentNode) {\n existingList.replaceWith(container)\n } else {\n // Fallback: append to discovery list container\n const listContainer = document.getElementById('discoveredList')\n if (listContainer) {\n listContainer.appendChild(container)\n } else {\n console.error('[SwitchBot][Discovery] render: discoveredList container not found in DOM (fallback)')\n toastError('Discovery UI error: device list container missing. Please reload the page.')\n }\n }\n\n // Only update the enabled/disabled state of batch action buttons (created in ensureDiscoveryControls)\n function updateBatchButtonStates() {\n // These buttons are created in ensureDiscoveryControls and should have unique IDs\n const addSelectedBtn = document.getElementById('addSelectedBtn') as HTMLButtonElement | null\n const enableSelectedBtn = document.getElementById('enableSelectedBtn') as HTMLButtonElement | null\n const disableSelectedBtn = document.getElementById('disableSelectedBtn') as HTMLButtonElement | null\n const hasSelection = selectedIds.size > 0\n if (addSelectedBtn) {\n addSelectedBtn.disabled = !hasSelection\n }\n if (enableSelectedBtn) {\n enableSelectedBtn.disabled = !hasSelection\n }\n if (disableSelectedBtn) {\n disableSelectedBtn.disabled = !hasSelection\n }\n }\n window.removeEventListener('discovery-selection-changed', updateBatchButtonStates)\n window.addEventListener('discovery-selection-changed', updateBatchButtonStates)\n // Initial state update\n updateBatchButtonStates()\n\n // Update status with count\n const status = document.getElementById('discoverStatus')\n if (status) {\n const totalCount = allDevices.length\n const filteredCount = visibleDevices.length\n status.textContent = filteredCount === totalCount\n ? `Found ${totalCount} device(s)`\n : `Showing ${filteredCount} of ${totalCount} device(s)`\n }\n}\n\nexport async function addDeviceToConfig(device: any): Promise<void> {\n const { addDeviceToConfig: addDevice } = await import('./devices.js')\n await addDevice(device)\n}\n", "export { DEVICE_TYPES } from '../../../device-types.js'\n", "import { DEVICE_TYPES } from './constants.js'\nimport { uiLog } from './logger.js'\n\nexport type ImportDiscoveredDeviceResult = {\n configDeviceName: string\n configDeviceType: string\n address?: string\n connectionPreference?: string\n room?: string\n encryptionKey?: string\n keyId?: string\n refreshRate?: number\n blePollingEnabled?: boolean\n blePollIntervalMs?: number\n} | null\n\nexport async function importDiscoveredDevice(device: any): Promise<ImportDiscoveredDeviceResult> {\n // --- OpenAPI Polling Interval (refreshRate) ---\n const openApiRefreshLabel = document.createElement('label')\n openApiRefreshLabel.textContent = 'OpenAPI Polling Interval (seconds)'\n openApiRefreshLabel.style.display = 'block'\n openApiRefreshLabel.style.marginBottom = '6px'\n openApiRefreshLabel.style.fontWeight = '500'\n openApiRefreshLabel.style.fontSize = '12px'\n openApiRefreshLabel.style.color = '#6b7280'\n openApiRefreshLabel.title = 'How often to poll this device via OpenAPI for status (in seconds). Overrides platform value if set. Default: 300 (5 minutes). Minimum: 30.'\n\n const openApiRefreshInput = document.createElement('input')\n openApiRefreshInput.type = 'number'\n openApiRefreshInput.value = device.refreshRate || 300\n openApiRefreshInput.min = '30'\n openApiRefreshInput.step = '1'\n openApiRefreshInput.style.width = '100%'\n openApiRefreshInput.style.marginBottom = '12px'\n openApiRefreshInput.style.padding = '8px 10px'\n openApiRefreshInput.style.borderRadius = '6px'\n openApiRefreshInput.style.fontSize = '14px'\n openApiRefreshInput.style.boxSizing = 'border-box'\n return new Promise((resolve) => {\n const div = document.createElement('div')\n div.style.position = 'fixed'\n div.style.top = '0'\n div.style.left = '0'\n div.style.width = '100%'\n div.style.height = '100%'\n div.style.background = 'rgba(0,0,0,0.7)'\n div.style.display = 'flex'\n div.style.alignItems = 'center'\n div.style.justifyContent = 'center'\n div.style.zIndex = '9999'\n\n const modal = document.createElement('div')\n modal.style.background = getComputedStyle(document.body).backgroundColor\n modal.style.color = getComputedStyle(document.body).color\n modal.style.padding = '0'\n modal.style.borderRadius = '10px'\n modal.style.minWidth = '440px'\n modal.style.maxWidth = '90vw'\n modal.style.boxShadow = '0 8px 32px rgba(0,0,0,0.35)'\n modal.style.overflow = 'hidden'\n modal.style.borderTop = '3px solid var(--switchbot-red, #ef4444)'\n\n const title = document.createElement('h3')\n title.textContent = 'Import Discovered Device'\n title.style.marginTop = '0'\n title.style.marginBottom = '16px'\n title.style.padding = '20px 20px 0'\n title.style.fontSize = '18px'\n title.style.fontWeight = '600'\n title.style.color = 'var(--switchbot-red, #ef4444)'\n title.style.letterSpacing = '-0.02em'\n\n const contentDiv = document.createElement('div')\n contentDiv.style.padding = '0 20px 20px'\n\n const nameLabel = document.createElement('label')\n nameLabel.textContent = 'Device Name'\n nameLabel.style.display = 'block'\n nameLabel.style.marginBottom = '6px'\n nameLabel.style.fontWeight = '500'\n nameLabel.style.fontSize = '12px'\n nameLabel.style.color = '#6b7280'\n\n const nameInput = document.createElement('input')\n nameInput.type = 'text'\n // Never allow 'undefined' as a name\n let safeName = device.name\n if (!safeName || safeName === 'undefined') {\n safeName = device.id || ''\n }\n nameInput.value = safeName\n nameInput.style.width = '100%'\n nameInput.style.marginBottom = '12px'\n nameInput.style.padding = '8px 10px'\n nameInput.style.borderRadius = '6px'\n nameInput.style.fontSize = '14px'\n nameInput.style.boxSizing = 'border-box'\n\n // --- Device Type Select ---\n const typeLabel = document.createElement('label')\n typeLabel.textContent = 'Config Device Type'\n typeLabel.style.display = 'block'\n typeLabel.style.marginBottom = '6px'\n typeLabel.style.fontWeight = '500'\n typeLabel.style.fontSize = '12px'\n typeLabel.style.color = '#6b7280'\n\n const typeSelect = document.createElement('select')\n typeSelect.style.width = '100%'\n typeSelect.style.padding = '8px 10px'\n typeSelect.style.marginBottom = '12px'\n typeSelect.style.borderRadius = '6px'\n typeSelect.style.fontSize = '14px'\n typeSelect.style.background = getComputedStyle(nameInput).background\n typeSelect.style.color = getComputedStyle(nameInput).color\n typeSelect.style.border = getComputedStyle(nameInput).border\n typeSelect.style.boxSizing = 'border-box'\n\n Object.keys(DEVICE_TYPES).forEach((categoryName) => {\n const optgroup = document.createElement('optgroup')\n optgroup.label = categoryName\n DEVICE_TYPES[categoryName as keyof typeof DEVICE_TYPES].forEach((deviceType) => {\n const opt = document.createElement('option')\n opt.value = deviceType\n opt.text = deviceType\n const detectedType = (device.type || '').toLowerCase()\n opt.selected = deviceType.toLowerCase() === detectedType\n optgroup.appendChild(opt)\n })\n typeSelect.appendChild(optgroup)\n })\n\n // --- Connection Preference ---\n const connectionPrefLabel = document.createElement('label')\n connectionPrefLabel.textContent = 'Connection Preference'\n connectionPrefLabel.style.display = 'block'\n connectionPrefLabel.style.marginBottom = '6px'\n connectionPrefLabel.style.fontWeight = '500'\n connectionPrefLabel.style.fontSize = '12px'\n connectionPrefLabel.style.color = '#6b7280'\n\n const connectionPrefSelect = document.createElement('select')\n connectionPrefSelect.style.width = '100%'\n connectionPrefSelect.style.marginBottom = '12px'\n connectionPrefSelect.style.padding = '8px 10px'\n connectionPrefSelect.style.borderRadius = '6px'\n connectionPrefSelect.style.fontSize = '14px'\n connectionPrefSelect.style.boxSizing = 'border-box';\n ['auto', 'ble', 'openapi'].forEach((val) => {\n const opt = document.createElement('option')\n opt.value = val\n opt.text = val.charAt(0).toUpperCase() + val.slice(1)\n opt.selected = (device.connectionPreference || 'auto') === val\n connectionPrefSelect.appendChild(opt)\n })\n\n // --- Room ---\n const roomLabel = document.createElement('label')\n roomLabel.textContent = 'Room'\n roomLabel.style.display = 'block'\n roomLabel.style.marginBottom = '6px'\n roomLabel.style.fontWeight = '500'\n roomLabel.style.fontSize = '12px'\n roomLabel.style.color = '#6b7280'\n\n const roomInput = document.createElement('input')\n roomInput.type = 'text'\n roomInput.value = device.room || ''\n roomInput.placeholder = 'Optional room/location metadata'\n roomInput.style.width = '100%'\n roomInput.style.marginBottom = '12px'\n roomInput.style.padding = '8px 10px'\n roomInput.style.borderRadius = '6px'\n roomInput.style.fontSize = '14px'\n roomInput.style.boxSizing = 'border-box'\n\n // --- BLE MAC Address ---\n const macLabel = document.createElement('label')\n macLabel.textContent = 'BLE MAC Address (optional)'\n macLabel.style.display = 'block'\n macLabel.style.marginBottom = '6px'\n macLabel.style.fontWeight = '500'\n macLabel.style.fontSize = '12px'\n macLabel.style.color = '#6b7280'\n\n const macInput = document.createElement('input')\n macInput.type = 'text'\n macInput.value = device.address || ''\n macInput.placeholder = 'AA:BB:CC:DD:EE:FF'\n macInput.style.width = '100%'\n macInput.style.marginBottom = '12px'\n macInput.style.padding = '8px 10px'\n macInput.style.borderRadius = '6px'\n macInput.style.fontSize = '14px'\n macInput.style.boxSizing = 'border-box'\n\n // --- Encryption Key ---\n const encryptionKeyLabel = document.createElement('label')\n encryptionKeyLabel.textContent = 'BLE Encryption Key (optional)'\n encryptionKeyLabel.style.display = 'block'\n encryptionKeyLabel.style.marginBottom = '6px'\n encryptionKeyLabel.style.fontWeight = '500'\n encryptionKeyLabel.style.fontSize = '12px'\n encryptionKeyLabel.style.color = '#6b7280'\n\n const encryptionKeyInput = document.createElement('input')\n encryptionKeyInput.type = 'password'\n encryptionKeyInput.value = device.encryptionKey || ''\n encryptionKeyInput.placeholder = 'Paste device BLE encryption key'\n encryptionKeyInput.style.width = '100%'\n encryptionKeyInput.style.marginBottom = '12px'\n encryptionKeyInput.style.padding = '8px 10px'\n encryptionKeyInput.style.borderRadius = '6px'\n encryptionKeyInput.style.fontSize = '14px'\n encryptionKeyInput.style.boxSizing = 'border-box'\n\n // --- Key ID ---\n const keyIdLabel = document.createElement('label')\n keyIdLabel.textContent = 'BLE Key ID (optional)'\n keyIdLabel.style.display = 'block'\n keyIdLabel.style.marginBottom = '6px'\n keyIdLabel.style.fontWeight = '500'\n keyIdLabel.style.fontSize = '12px'\n keyIdLabel.style.color = '#6b7280'\n\n const keyIdInput = document.createElement('input')\n keyIdInput.type = 'text'\n keyIdInput.value = device.keyId || ''\n keyIdInput.placeholder = 'e.g. ff'\n keyIdInput.style.width = '100%'\n keyIdInput.style.marginBottom = '12px'\n keyIdInput.style.padding = '8px 10px'\n keyIdInput.style.borderRadius = '6px'\n keyIdInput.style.fontSize = '14px'\n keyIdInput.style.boxSizing = 'border-box'\n\n // --- BLE Polling Enabled ---\n const blePollingEnabledLabel = document.createElement('label')\n blePollingEnabledLabel.textContent = 'Enable BLE Polling Fallback'\n blePollingEnabledLabel.style.display = 'block'\n blePollingEnabledLabel.style.marginBottom = '6px'\n blePollingEnabledLabel.style.fontWeight = '500'\n blePollingEnabledLabel.style.fontSize = '12px'\n blePollingEnabledLabel.style.color = '#6b7280'\n\n const blePollingEnabledInput = document.createElement('input')\n blePollingEnabledInput.type = 'checkbox'\n blePollingEnabledInput.checked = device.blePollingEnabled !== false // default true\n blePollingEnabledInput.style.marginRight = '8px'\n blePollingEnabledInput.style.marginBottom = '12px'\n\n // --- BLE Poll Interval ---\n const blePollIntervalLabel = document.createElement('label')\n blePollIntervalLabel.textContent = 'BLE Polling Interval (ms)'\n blePollIntervalLabel.style.display = 'block'\n blePollIntervalLabel.style.marginBottom = '6px'\n blePollIntervalLabel.style.fontWeight = '500'\n blePollIntervalLabel.style.fontSize = '12px'\n blePollIntervalLabel.style.color = '#6b7280'\n\n const blePollIntervalInput = document.createElement('input')\n blePollIntervalInput.type = 'number'\n blePollIntervalInput.value = device.blePollIntervalMs || 600000\n blePollIntervalInput.min = '60000'\n blePollIntervalInput.step = '1000'\n blePollIntervalInput.style.width = '100%'\n blePollIntervalInput.style.marginBottom = '12px'\n blePollIntervalInput.style.padding = '8px 10px'\n blePollIntervalInput.style.borderRadius = '6px'\n blePollIntervalInput.style.fontSize = '14px'\n blePollIntervalInput.style.boxSizing = 'border-box'\n\n const buttons = document.createElement('div')\n buttons.style.display = 'flex'\n buttons.style.gap = '10px'\n buttons.style.justifyContent = 'flex-end'\n buttons.style.marginTop = '18px'\n buttons.style.paddingTop = '18px'\n buttons.style.borderTop = '1px solid rgba(0, 0, 0, 0.08)'\n\n const cancelBtn = document.createElement('button')\n cancelBtn.textContent = 'Cancel'\n cancelBtn.className = 'secondary'\n cancelBtn.style.background = '#6b7280'\n cancelBtn.style.padding = '8px 16px'\n cancelBtn.style.fontSize = '13px'\n\n const importBtn = document.createElement('button')\n importBtn.textContent = 'Add to Config'\n importBtn.style.background = 'var(--switchbot-red, #ef4444)'\n importBtn.style.padding = '8px 20px'\n importBtn.style.fontSize = '13px'\n\n const cleanup = (result: ImportDiscoveredDeviceResult) => {\n div.remove()\n resolve(result)\n }\n\n cancelBtn.onclick = () => cleanup(null)\n importBtn.onclick = () => {\n // Never allow 'undefined' as a name\n let finalName = nameInput.value\n if (!finalName || finalName === 'undefined') {\n finalName = device.id || ''\n }\n cleanup({\n configDeviceName: finalName,\n configDeviceType: typeSelect.value || device.type,\n address: macInput.value || undefined,\n connectionPreference: connectionPrefSelect.value || undefined,\n room: roomInput.value || undefined,\n encryptionKey: encryptionKeyInput.value || undefined,\n keyId: keyIdInput.value || undefined,\n refreshRate: Number(openApiRefreshInput.value) || 300,\n blePollingEnabled: blePollingEnabledInput.checked,\n blePollIntervalMs: Number(blePollIntervalInput.value) || 600000,\n })\n }\n\n div.addEventListener('click', (event) => {\n if (event.target === div) {\n cleanup(null)\n }\n })\n\n buttons.appendChild(cancelBtn)\n buttons.appendChild(importBtn)\n contentDiv.appendChild(nameLabel)\n contentDiv.appendChild(nameInput)\n contentDiv.appendChild(typeLabel)\n contentDiv.appendChild(typeSelect)\n contentDiv.appendChild(connectionPrefLabel)\n contentDiv.appendChild(connectionPrefSelect)\n contentDiv.appendChild(roomLabel)\n contentDiv.appendChild(roomInput)\n contentDiv.appendChild(macLabel)\n contentDiv.appendChild(macInput)\n contentDiv.appendChild(encryptionKeyLabel)\n contentDiv.appendChild(encryptionKeyInput)\n contentDiv.appendChild(keyIdLabel)\n contentDiv.appendChild(keyIdInput)\n\n contentDiv.appendChild(openApiRefreshLabel)\n contentDiv.appendChild(openApiRefreshInput)\n contentDiv.appendChild(blePollingEnabledLabel)\n contentDiv.appendChild(blePollingEnabledInput)\n contentDiv.appendChild(blePollIntervalLabel)\n contentDiv.appendChild(blePollIntervalInput)\n contentDiv.appendChild(buttons)\n\n modal.appendChild(title)\n modal.appendChild(contentDiv)\n div.appendChild(modal)\n document.body.appendChild(div)\n nameInput.focus()\n })\n}\n\nexport async function editDevice(device: any): Promise<void> {\n // --- OpenAPI Polling Interval (refreshRate) ---\n const openApiRefreshLabel = document.createElement('label')\n openApiRefreshLabel.textContent = 'OpenAPI Polling Interval (seconds)'\n openApiRefreshLabel.style.display = 'block'\n openApiRefreshLabel.style.marginBottom = '6px'\n openApiRefreshLabel.style.fontWeight = '500'\n openApiRefreshLabel.style.fontSize = '12px'\n openApiRefreshLabel.style.color = '#6b7280'\n openApiRefreshLabel.title = 'How often to poll this device via OpenAPI for status (in seconds). Overrides platform value if set. Default: 300 (5 minutes). Minimum: 30.'\n\n const openApiRefreshInput = document.createElement('input')\n openApiRefreshInput.type = 'number'\n openApiRefreshInput.value = device.refreshRate || 300\n openApiRefreshInput.min = '30'\n openApiRefreshInput.step = '1'\n openApiRefreshInput.style.width = '100%'\n openApiRefreshInput.style.marginBottom = '12px'\n openApiRefreshInput.style.padding = '8px 10px'\n openApiRefreshInput.style.borderRadius = '6px'\n openApiRefreshInput.style.fontSize = '14px'\n openApiRefreshInput.style.boxSizing = 'border-box'\n // --- Device Type Select ---\n // --- BLE Polling Enabled ---\n const blePollingEnabledLabel = document.createElement('label')\n blePollingEnabledLabel.textContent = 'Enable BLE Polling Fallback'\n blePollingEnabledLabel.style.display = 'block'\n blePollingEnabledLabel.style.marginBottom = '6px'\n blePollingEnabledLabel.style.fontWeight = '500'\n blePollingEnabledLabel.style.fontSize = '12px'\n blePollingEnabledLabel.style.color = '#6b7280'\n\n const blePollingEnabledInput = document.createElement('input')\n blePollingEnabledInput.type = 'checkbox'\n blePollingEnabledInput.checked = device.blePollingEnabled !== false // default true\n blePollingEnabledInput.style.marginRight = '8px'\n blePollingEnabledInput.style.marginBottom = '12px'\n\n // --- BLE Poll Interval ---\n const blePollIntervalLabel = document.createElement('label')\n blePollIntervalLabel.textContent = 'BLE Polling Interval (ms)'\n blePollIntervalLabel.style.display = 'block'\n blePollIntervalLabel.style.marginBottom = '6px'\n blePollIntervalLabel.style.fontWeight = '500'\n blePollIntervalLabel.style.fontSize = '12px'\n blePollIntervalLabel.style.color = '#6b7280'\n\n const blePollIntervalInput = document.createElement('input')\n blePollIntervalInput.type = 'number'\n blePollIntervalInput.value = device.blePollIntervalMs || 600000\n blePollIntervalInput.min = '60000'\n blePollIntervalInput.step = '1000'\n blePollIntervalInput.style.width = '100%'\n blePollIntervalInput.style.marginBottom = '12px'\n blePollIntervalInput.style.padding = '8px 10px'\n blePollIntervalInput.style.borderRadius = '6px'\n blePollIntervalInput.style.fontSize = '14px'\n blePollIntervalInput.style.boxSizing = 'border-box'\n const typeLabel = document.createElement('label')\n typeLabel.textContent = 'Config Device Type'\n typeLabel.style.display = 'block'\n typeLabel.style.marginBottom = '6px'\n typeLabel.style.fontWeight = '500'\n typeLabel.style.fontSize = '12px'\n typeLabel.style.color = '#6b7280'\n\n const typeSelect = document.createElement('select')\n typeSelect.style.width = '100%'\n typeSelect.style.padding = '8px 10px'\n typeSelect.style.marginBottom = '12px'\n typeSelect.style.borderRadius = '6px'\n typeSelect.style.fontSize = '14px'\n typeSelect.style.background = getComputedStyle(document.body).backgroundColor\n typeSelect.style.color = getComputedStyle(document.body).color\n typeSelect.style.border = '1px solid #ccc'\n typeSelect.style.boxSizing = 'border-box'\n\n // Add option groups with all API device types\n Object.keys(DEVICE_TYPES).forEach((categoryName) => {\n const optgroup = document.createElement('optgroup')\n optgroup.label = categoryName\n DEVICE_TYPES[categoryName as keyof typeof DEVICE_TYPES].forEach((deviceType) => {\n const opt = document.createElement('option')\n opt.value = deviceType\n opt.text = deviceType\n // Select current configDeviceType if set, otherwise match API deviceType\n const currentType = device.configDeviceType || device.deviceType || device.type || ''\n opt.selected = currentType === deviceType\n optgroup.appendChild(opt)\n })\n typeSelect.appendChild(optgroup)\n })\n\n // Create modal dialog for editing\n const div = document.createElement('div')\n div.style.position = 'fixed'\n div.style.top = '0'\n div.style.left = '0'\n div.style.width = '100%'\n div.style.height = '100%'\n div.style.background = 'rgba(0,0,0,0.7)'\n div.style.display = 'flex'\n div.style.alignItems = 'center'\n div.style.justifyContent = 'center'\n div.style.zIndex = '9999'\n\n const modal = document.createElement('div')\n modal.style.background = getComputedStyle(document.body).backgroundColor\n modal.style.color = getComputedStyle(document.body).color\n modal.style.padding = '0'\n modal.style.borderRadius = '10px'\n modal.style.minWidth = '440px'\n modal.style.maxWidth = '90vw'\n modal.style.boxShadow = '0 8px 32px rgba(0,0,0,0.35)'\n modal.style.overflow = 'hidden'\n modal.style.borderTop = '3px solid var(--switchbot-red, #ef4444)'\n\n const title = document.createElement('h3')\n title.textContent = 'Edit Device'\n title.style.marginTop = '0'\n title.style.marginBottom = '16px'\n title.style.padding = '20px 20px 0'\n title.style.fontSize = '18px'\n title.style.fontWeight = '600'\n title.style.color = 'var(--switchbot-red, #ef4444)'\n title.style.letterSpacing = '-0.02em'\n\n const contentDiv = document.createElement('div')\n contentDiv.style.padding = '0 20px 20px'\n\n const nameLabel = document.createElement('label')\n nameLabel.textContent = 'Device Name'\n nameLabel.style.display = 'block'\n nameLabel.style.marginBottom = '6px'\n nameLabel.style.fontWeight = '500'\n nameLabel.style.fontSize = '12px'\n nameLabel.style.color = '#6b7280'\n\n const nameInput = document.createElement('input')\n nameInput.type = 'text'\n nameInput.value = device.name || device.id\n nameInput.style.width = '100%'\n nameInput.style.marginBottom = '12px'\n nameInput.style.padding = '8px 10px'\n nameInput.style.borderRadius = '6px'\n nameInput.style.fontSize = '14px'\n nameInput.style.boxSizing = 'border-box'\n nameInput.style.transition = 'border-color 0.2s ease'\n\n // Read-only API device type field\n const apiTypeLabel = document.createElement('label')\n apiTypeLabel.textContent = 'Device Type (API - Read Only)'\n apiTypeLabel.style.display = 'block'\n apiTypeLabel.style.marginBottom = '6px'\n apiTypeLabel.style.fontWeight = '500'\n apiTypeLabel.style.fontSize = '12px'\n apiTypeLabel.style.color = '#6b7280'\n\n const apiTypeInput = document.createElement('input')\n apiTypeInput.type = 'text'\n apiTypeInput.value = device.deviceType || device.type || 'Unknown'\n apiTypeInput.readOnly = true\n apiTypeInput.style.width = '100%'\n apiTypeInput.style.marginBottom = '12px'\n apiTypeInput.style.padding = '8px 10px'\n apiTypeInput.style.borderRadius = '6px'\n apiTypeInput.style.fontSize = '13px'\n apiTypeInput.style.opacity = '0.6'\n apiTypeInput.style.cursor = 'not-allowed'\n apiTypeInput.style.boxSizing = 'border-box'\n apiTypeInput.style.backgroundColor = '#f9fafb'\n\n // Editable config device type dropdown\n\n // Add option groups with all API device types\n Object.keys(DEVICE_TYPES).forEach((categoryName) => {\n const optgroup = document.createElement('optgroup')\n optgroup.label = categoryName\n\n DEVICE_TYPES[categoryName as keyof typeof DEVICE_TYPES].forEach((deviceType) => {\n const opt = document.createElement('option')\n opt.value = deviceType\n opt.text = deviceType\n\n // Select current configDeviceType if set, otherwise match API deviceType\n const currentType = device.configDeviceType || device.deviceType || device.type || ''\n opt.selected = currentType === deviceType\n optgroup.appendChild(opt)\n })\n\n typeSelect.appendChild(optgroup)\n })\n\n // --- Connection Preference ---\n const connectionPrefLabel = document.createElement('label')\n connectionPrefLabel.textContent = 'Connection Preference'\n connectionPrefLabel.style.display = 'block'\n connectionPrefLabel.style.marginBottom = '6px'\n connectionPrefLabel.style.fontWeight = '500'\n connectionPrefLabel.style.fontSize = '12px'\n connectionPrefLabel.style.color = '#6b7280'\n\n const connectionPrefSelect = document.createElement('select')\n connectionPrefSelect.style.width = '100%'\n connectionPrefSelect.style.marginBottom = '12px'\n connectionPrefSelect.style.padding = '8px 10px'\n connectionPrefSelect.style.borderRadius = '6px'\n connectionPrefSelect.style.fontSize = '14px'\n connectionPrefSelect.style.boxSizing = 'border-box'\n ;['auto', 'ble', 'openapi'].forEach((val) => {\n const opt = document.createElement('option')\n opt.value = val\n opt.text = val.charAt(0).toUpperCase() + val.slice(1)\n opt.selected = (device.connectionPreference || 'auto') === val\n connectionPrefSelect.appendChild(opt)\n })\n\n // --- Room ---\n const roomLabel = document.createElement('label')\n roomLabel.textContent = 'Room'\n roomLabel.style.display = 'block'\n roomLabel.style.marginBottom = '6px'\n roomLabel.style.fontWeight = '500'\n roomLabel.style.fontSize = '12px'\n roomLabel.style.color = '#6b7280'\n\n const roomInput = document.createElement('input')\n roomInput.type = 'text'\n roomInput.value = device.room || ''\n roomInput.placeholder = 'Optional room/location metadata'\n roomInput.style.width = '100%'\n roomInput.style.marginBottom = '12px'\n roomInput.style.padding = '8px 10px'\n roomInput.style.borderRadius = '6px'\n roomInput.style.fontSize = '14px'\n roomInput.style.boxSizing = 'border-box'\n\n // --- Encryption Key ---\n const encryptionKeyLabel = document.createElement('label')\n encryptionKeyLabel.textContent = 'BLE Encryption Key (optional)'\n encryptionKeyLabel.style.display = 'block'\n encryptionKeyLabel.style.marginBottom = '6px'\n encryptionKeyLabel.style.fontWeight = '500'\n encryptionKeyLabel.style.fontSize = '12px'\n encryptionKeyLabel.style.color = '#6b7280'\n\n const encryptionKeyInput = document.createElement('input')\n encryptionKeyInput.type = 'password'\n encryptionKeyInput.value = device.encryptionKey || ''\n encryptionKeyInput.placeholder = 'Paste device BLE encryption key'\n encryptionKeyInput.style.width = '100%'\n encryptionKeyInput.style.marginBottom = '12px'\n encryptionKeyInput.style.padding = '8px 10px'\n encryptionKeyInput.style.borderRadius = '6px'\n encryptionKeyInput.style.fontSize = '14px'\n encryptionKeyInput.style.boxSizing = 'border-box'\n\n // --- Key ID ---\n const keyIdLabel = document.createElement('label')\n keyIdLabel.textContent = 'BLE Key ID (optional)'\n keyIdLabel.style.display = 'block'\n keyIdLabel.style.marginBottom = '6px'\n keyIdLabel.style.fontWeight = '500'\n keyIdLabel.style.fontSize = '12px'\n keyIdLabel.style.color = '#6b7280'\n\n const keyIdInput = document.createElement('input')\n keyIdInput.type = 'text'\n keyIdInput.value = device.keyId || ''\n keyIdInput.placeholder = 'e.g. ff'\n keyIdInput.style.width = '100%'\n keyIdInput.style.marginBottom = '12px'\n keyIdInput.style.padding = '8px 10px'\n keyIdInput.style.borderRadius = '6px'\n keyIdInput.style.fontSize = '14px'\n keyIdInput.style.boxSizing = 'border-box'\n\n const errorMessage = document.createElement('div')\n errorMessage.style.color = 'var(--switchbot-red, #ef4444)'\n errorMessage.style.marginBottom = '12px'\n errorMessage.style.fontSize = '12px'\n errorMessage.style.display = 'none'\n errorMessage.style.padding = '8px 10px'\n errorMessage.style.background = 'var(--switchbot-red-light, #fee2e2)'\n errorMessage.style.borderRadius = '6px'\n errorMessage.style.fontWeight = '500'\n\n const buttons = document.createElement('div')\n buttons.style.display = 'flex'\n buttons.style.gap = '10px'\n buttons.style.justifyContent = 'flex-end'\n buttons.style.marginTop = '18px'\n buttons.style.paddingTop = '18px'\n buttons.style.borderTop = '1px solid rgba(0, 0, 0, 0.08)'\n\n const cancelBtn = document.createElement('button')\n cancelBtn.textContent = 'Cancel'\n cancelBtn.className = 'secondary'\n cancelBtn.style.background = '#6b7280'\n cancelBtn.style.padding = '8px 16px'\n cancelBtn.style.fontSize = '13px'\n cancelBtn.onclick = () => div.remove()\n\n const saveBtn = document.createElement('button')\n saveBtn.textContent = 'Save'\n saveBtn.style.background = 'var(--switchbot-red, #ef4444)'\n saveBtn.style.padding = '8px 20px'\n saveBtn.style.fontSize = '13px'\n saveBtn.onclick = async () => {\n try {\n const { updateDevice, syncParentPluginConfigFromDisk, fetchDevices } = await import('./api.js')\n const { renderDeviceList } = await import('./render.js')\n\n const params = {\n deviceId: device.id,\n configDeviceName: nameInput.value || undefined,\n configDeviceType: typeSelect.value,\n connectionPreference: connectionPrefSelect.value,\n room: roomInput.value || undefined,\n encryptionKey: encryptionKeyInput.value || undefined,\n keyId: keyIdInput.value || undefined,\n refreshRate: Number(openApiRefreshInput.value) || 300,\n blePollingEnabled: blePollingEnabledInput.checked,\n blePollIntervalMs: Number(blePollIntervalInput.value) || 600000,\n }\n\n // Only include defined properties in the options object\n const options: any = {}\n if (params.connectionPreference !== undefined) {\n options.connectionPreference = params.connectionPreference\n }\n if (params.room !== undefined) {\n options.room = params.room\n }\n if (params.encryptionKey !== undefined) {\n options.encryptionKey = params.encryptionKey\n }\n if (params.keyId !== undefined) {\n options.keyId = params.keyId\n }\n if (params.refreshRate !== undefined) {\n options.refreshRate = params.refreshRate\n }\n if (params.blePollingEnabled !== undefined) {\n options.blePollingEnabled = params.blePollingEnabled\n }\n if (params.blePollIntervalMs !== undefined) {\n options.blePollIntervalMs = params.blePollIntervalMs\n }\n\n await updateDevice(\n params.deviceId,\n params.configDeviceName,\n params.configDeviceType,\n options,\n )\n await syncParentPluginConfigFromDisk()\n contentDiv.appendChild(openApiRefreshLabel)\n contentDiv.appendChild(openApiRefreshInput)\n\n // Refresh device list\n uiLog.info('[Edit Device] Refreshing device list after update')\n const list = await fetchDevices()\n renderDeviceList(list)\n div.remove()\n } catch (e) {\n uiLog.error('Update error:', e)\n errorMessage.textContent = `Error: ${e instanceof Error ? e.message : 'Failed to update device'}`\n errorMessage.style.display = 'block'\n }\n }\n\n buttons.appendChild(cancelBtn)\n buttons.appendChild(saveBtn)\n\n contentDiv.appendChild(nameLabel)\n contentDiv.appendChild(nameInput)\n contentDiv.appendChild(apiTypeLabel)\n contentDiv.appendChild(apiTypeInput)\n contentDiv.appendChild(typeLabel)\n contentDiv.appendChild(typeSelect)\n contentDiv.appendChild(connectionPrefLabel)\n contentDiv.appendChild(connectionPrefSelect)\n contentDiv.appendChild(roomLabel)\n contentDiv.appendChild(roomInput)\n contentDiv.appendChild(encryptionKeyLabel)\n contentDiv.appendChild(encryptionKeyInput)\n contentDiv.appendChild(keyIdLabel)\n contentDiv.appendChild(keyIdInput)\n\n // Insert BLE polling fields before error/buttons\n contentDiv.appendChild(openApiRefreshLabel)\n contentDiv.appendChild(openApiRefreshInput)\n contentDiv.appendChild(blePollingEnabledLabel)\n contentDiv.appendChild(blePollingEnabledInput)\n contentDiv.appendChild(blePollIntervalLabel)\n contentDiv.appendChild(blePollIntervalInput)\n contentDiv.appendChild(errorMessage)\n contentDiv.appendChild(buttons)\n\n modal.appendChild(title)\n modal.appendChild(contentDiv)\n\n div.appendChild(modal)\n document.body.appendChild(div)\n nameInput.focus()\n}\n", "import { deleteAllDevices as apiDeleteAllDevices, deleteDevice as apiDeleteDevice, fetchDevices, syncParentPluginConfigFromDisk } from './api.js'\nimport { uiLog } from './logger.js'\nimport { hideBusyUi, showBusyUi } from './modal.js'\nimport { renderDeviceList } from './render.js'\nimport { toastError, toastSuccess, toastWarning } from './toast.js'\n\nasync function confirmDeleteDialog(deviceNameOrId: string): Promise<boolean> {\n return new Promise((resolve) => {\n const overlay = document.createElement('div')\n overlay.style.position = 'fixed'\n overlay.style.top = '0'\n overlay.style.left = '0'\n overlay.style.width = '100%'\n overlay.style.height = '100%'\n overlay.style.background = 'rgba(0,0,0,0.6)'\n overlay.style.display = 'flex'\n overlay.style.alignItems = 'center'\n overlay.style.justifyContent = 'center'\n overlay.style.zIndex = '10000'\n\n const modal = document.createElement('div')\n modal.style.background = getComputedStyle(document.body).backgroundColor\n modal.style.color = getComputedStyle(document.body).color\n modal.style.padding = '20px'\n modal.style.borderRadius = '10px'\n modal.style.minWidth = '340px'\n modal.style.maxWidth = '90vw'\n modal.style.boxShadow = '0 12px 40px rgba(0,0,0,0.35)'\n\n const title = document.createElement('h3')\n title.textContent = 'Delete Device'\n title.style.margin = '0 0 10px 0'\n\n const message = document.createElement('p')\n message.textContent = `Remove \"${deviceNameOrId}\" from configuration?`\n message.style.margin = '0 0 16px 0'\n\n const actions = document.createElement('div')\n actions.style.display = 'flex'\n actions.style.justifyContent = 'flex-end'\n actions.style.gap = '8px'\n\n const cancelBtn = document.createElement('button')\n cancelBtn.textContent = 'Cancel'\n cancelBtn.style.background = '#6b7280'\n\n const deleteBtn = document.createElement('button')\n deleteBtn.textContent = 'Delete'\n deleteBtn.style.background = '#ef4444'\n\n const cleanup = (result: boolean) => {\n overlay.remove()\n resolve(result)\n }\n\n cancelBtn.onclick = () => cleanup(false)\n deleteBtn.onclick = () => cleanup(true)\n overlay.addEventListener('click', (event) => {\n if (event.target === overlay) {\n cleanup(false)\n }\n })\n\n actions.appendChild(cancelBtn)\n actions.appendChild(deleteBtn)\n modal.appendChild(title)\n modal.appendChild(message)\n modal.appendChild(actions)\n overlay.appendChild(modal)\n document.body.appendChild(overlay)\n })\n}\n\nexport async function deleteDeviceFromConfig(deviceId: string, deviceName: string): Promise<void> {\n uiLog.info('Delete button clicked for device:', deviceId, deviceName)\n\n const confirmed = await confirmDeleteDialog(deviceName || deviceId)\n if (!confirmed) {\n uiLog.info('Delete cancelled by user')\n return\n }\n\n try {\n showBusyUi()\n uiLog.info('Deleting device from config:', deviceId)\n const resp = await apiDeleteDevice(deviceId)\n uiLog.info('Delete response:', resp)\n\n uiLog.info('Syncing parent config from disk...')\n const synced = await syncParentPluginConfigFromDisk(true)\n if (!synced) {\n toastWarning('Device deleted, but configuration sync failed')\n }\n\n uiLog.info('Refreshing device list...')\n const list = await fetchDevices()\n uiLog.info('Rendering devices:', list.length)\n renderDeviceList(list)\n\n uiLog.info('\u2713 Device deleted successfully')\n toastSuccess(`Device \"${deviceName || deviceId}\" deleted successfully`)\n } catch (e) {\n uiLog.error('Delete error:', e)\n toastError(e instanceof Error ? e.message : 'Failed to delete device')\n } finally {\n hideBusyUi()\n }\n}\n\nasync function confirmDeleteAllDialog(deviceCount: number): Promise<boolean> {\n return new Promise((resolve) => {\n const overlay = document.createElement('div')\n overlay.style.position = 'fixed'\n overlay.style.top = '0'\n overlay.style.left = '0'\n overlay.style.width = '100%'\n overlay.style.height = '100%'\n overlay.style.background = 'rgba(0,0,0,0.7)'\n overlay.style.display = 'flex'\n overlay.style.alignItems = 'center'\n overlay.style.justifyContent = 'center'\n overlay.style.zIndex = '10000'\n\n const modal = document.createElement('div')\n modal.style.background = getComputedStyle(document.body).backgroundColor\n modal.style.color = getComputedStyle(document.body).color\n modal.style.padding = '20px'\n modal.style.borderRadius = '10px'\n modal.style.minWidth = '380px'\n modal.style.maxWidth = '90vw'\n modal.style.boxShadow = '0 12px 40px rgba(0,0,0,0.35)'\n modal.style.borderTop = '3px solid #ef4444'\n\n const title = document.createElement('h3')\n title.textContent = '\u26A0\uFE0F Remove All Devices'\n title.style.margin = '0 0 12px 0'\n title.style.color = '#ef4444'\n\n const message = document.createElement('p')\n message.innerHTML = `Are you sure you want to remove <strong>all ${deviceCount} device(s)</strong> from your configuration?<br><br>This action cannot be undone.`\n message.style.margin = '0 0 18px 0'\n message.style.lineHeight = '1.5'\n\n const actions = document.createElement('div')\n actions.style.display = 'flex'\n actions.style.justifyContent = 'flex-end'\n actions.style.gap = '10px'\n\n const cancelBtn = document.createElement('button')\n cancelBtn.textContent = 'Cancel'\n cancelBtn.style.background = '#6b7280'\n cancelBtn.style.padding = '8px 16px'\n cancelBtn.style.fontSize = '13px'\n cancelBtn.className = 'secondary'\n\n const deleteBtn = document.createElement('button')\n deleteBtn.textContent = 'Remove All'\n deleteBtn.style.background = '#ef4444'\n deleteBtn.style.padding = '8px 20px'\n deleteBtn.style.fontSize = '13px'\n\n const cleanup = (result: boolean) => {\n overlay.remove()\n resolve(result)\n }\n\n cancelBtn.onclick = () => cleanup(false)\n deleteBtn.onclick = () => cleanup(true)\n overlay.addEventListener('click', (event) => {\n if (event.target === overlay) {\n cleanup(false)\n }\n })\n\n actions.appendChild(cancelBtn)\n actions.appendChild(deleteBtn)\n modal.appendChild(title)\n modal.appendChild(message)\n modal.appendChild(actions)\n overlay.appendChild(modal)\n document.body.appendChild(overlay)\n })\n}\n\nexport async function deleteAllDevicesFromConfig(): Promise<void> {\n uiLog.info('Remove All Devices button clicked')\n\n try {\n // Get current device count\n const list = await fetchDevices()\n if (!list || list.length === 0) {\n toastWarning('No devices to remove')\n return\n }\n\n const confirmed = await confirmDeleteAllDialog(list.length)\n if (!confirmed) {\n uiLog.info('Remove all cancelled by user')\n return\n }\n\n showBusyUi()\n uiLog.info('Deleting all devices from config')\n const resp = await apiDeleteAllDevices()\n uiLog.info('Delete all response:', resp)\n\n uiLog.info('Syncing parent config from disk...')\n const synced = await syncParentPluginConfigFromDisk(true)\n if (!synced) {\n toastWarning('Devices deleted, but configuration sync failed')\n }\n\n uiLog.info('Refreshing device list...')\n const updatedList = await fetchDevices()\n uiLog.info('Rendering devices:', updatedList.length)\n renderDeviceList(updatedList)\n\n const deletedCount = resp?.deletedCount || 0\n uiLog.info(`\u2713 Removed ${deletedCount} device(s) successfully`)\n toastSuccess(`Removed ${deletedCount} device(s) successfully`)\n } catch (e) {\n uiLog.error('Delete all error:', e)\n toastError(e instanceof Error ? e.message : 'Failed to delete all devices')\n } finally {\n hideBusyUi()\n }\n}\n", "// Move regexes to module scope to avoid re-compilation on every call\n// import type { DEVICE_TYPES } from './constants.js' // Removed unused import\n\nconst SPACES_REGEX = /\\s/g\nconst CAMELCASE_REGEX = /([A-Z])/g\nconst FIRST_CHAR_REGEX = /^./\n\n/**\n * Get RSSI signal quality level and color based on dBm value\n * @param rssi Signal strength in dBm (typically -30 to -90)\n * @returns Object with quality level, color, and description\n */\nexport function getRssiSignalQuality(rssi: number | undefined): {\n level: 'excellent' | 'good' | 'fair' | 'poor' | 'unknown'\n color: string\n bgColor: string\n description: string\n bars: number\n} {\n if (!rssi || rssi === 0) {\n return {\n level: 'unknown',\n color: '#999',\n bgColor: '#f5f5f5',\n description: 'Signal strength unknown',\n bars: 0,\n }\n }\n\n const dbm = Math.floor(rssi)\n\n if (dbm > -60) {\n return {\n level: 'excellent',\n color: '#34a853',\n bgColor: '#e8f5e9',\n description: `Excellent (${dbm} dBm)`,\n bars: 4,\n }\n } else if (dbm > -75) {\n return {\n level: 'good',\n color: '#fbbc04',\n bgColor: '#fffde7',\n description: `Good (${dbm} dBm)`,\n bars: 3,\n }\n } else if (dbm > -85) {\n return {\n level: 'fair',\n color: '#ff9800',\n bgColor: '#fff3e0',\n description: `Fair (${dbm} dBm)`,\n bars: 2,\n }\n } else {\n return {\n level: 'poor',\n color: '#ea4335',\n bgColor: '#ffebee',\n description: `Poor (${dbm} dBm) - unreliable`,\n bars: 1,\n }\n }\n}\n\n/**\n * Create visual signal strength indicator bars\n * @param rssi Signal strength in dBm\n * @returns HTML element showing filled bars\n */\nexport function renderSignalBars(rssi: number | undefined): HTMLElement {\n const quality = getRssiSignalQuality(rssi)\n\n const container = document.createElement('span')\n container.style.display = 'inline-flex'\n container.style.gap = '2px'\n container.style.alignItems = 'center'\n container.style.marginLeft = '8px'\n container.style.fontSize = '12px'\n\n // Create 4 bars\n for (let i = 1; i <= 4; i++) {\n const bar = document.createElement('span')\n bar.style.height = `${i * 3}px`\n bar.style.width = '3px'\n bar.style.borderRadius = '1px'\n bar.style.border = `1px solid ${quality.color}`\n\n if (i <= quality.bars) {\n bar.style.backgroundColor = quality.color\n } else {\n bar.style.backgroundColor = 'transparent'\n }\n\n container.appendChild(bar)\n }\n\n // Add tooltip\n container.title = quality.description\n\n return container\n}\n\n/**\n * Create signal quality badge with color\n * @param rssi Signal strength in dBm\n * @returns HTML element showing quality level\n */\nexport function renderSignalQualityBadge(rssi: number | undefined): HTMLElement {\n const quality = getRssiSignalQuality(rssi)\n\n const badge = document.createElement('span')\n badge.textContent = quality.level.charAt(0).toUpperCase() + quality.level.slice(1)\n badge.style.cssText = `\n background: ${quality.color};\n color: white;\n padding: 2px 6px;\n border-radius: 3px;\n font-size: 10px;\n font-weight: 600;\n margin-left: 8px;\n `\n badge.title = quality.description\n\n return badge\n}\n\nexport function renderBadge(text: string, style: string): HTMLElement {\n const badge = document.createElement('span')\n badge.textContent = text\n badge.style.cssText = style\n return badge\n}\n\nexport function renderConnectionBadge(connectionType: string): HTMLElement | null {\n if (!connectionType) {\n return null\n }\n\n const badge = renderBadge(connectionType, '')\n\n if (connectionType === 'BLE') {\n badge.style.cssText\n = 'background: #4285f4; color: white; padding: 2px 6px; border-radius: 3px; font-size: 10px; font-weight: 600; margin-left: 8px;'\n } else if (connectionType === 'Both') {\n badge.style.cssText\n = 'background: #34a853; color: white; padding: 2px 6px; border-radius: 3px; font-size: 10px; font-weight: 600; margin-left: 8px;'\n } else {\n badge.style.cssText\n = 'background: #9e9e9e; color: white; padding: 2px 6px; border-radius: 3px; font-size: 10px; font-weight: 600; margin-left: 8px;'\n }\n\n return badge\n}\n\nexport function renderIRBadge(): HTMLElement {\n return renderBadge(\n 'IR',\n 'background: #ff6b35; color: white; padding: 2px 6px; border-radius: 3px; font-size: 10px; font-weight: 600; margin-left: 8px;',\n )\n}\n\nfunction normalizeId(value: any): string {\n return String(value ?? '').trim().toLowerCase()\n}\n\nfunction scrollToConfiguredDevice(deviceId: string): void {\n const normalizedId = normalizeId(deviceId)\n const target = document.querySelector(`[data-device-id=\"${normalizedId}\"]`) as HTMLElement | null\n if (!target) {\n return\n }\n\n target.scrollIntoView({ behavior: 'smooth', block: 'center' })\n const originalOutline = target.style.outline\n const originalBackground = target.style.background\n target.style.outline = '2px solid var(--switchbot-red, #ef4444)'\n target.style.background = 'rgba(239, 68, 68, 0.08)'\n setTimeout(() => {\n target.style.outline = originalOutline\n target.style.background = originalBackground\n }, 1800)\n}\n\nfunction createConnectionTestControls(device: any): HTMLElement {\n const controls = document.createElement('div')\n controls.style.display = 'inline-flex'\n controls.style.alignItems = 'center'\n controls.style.gap = '6px'\n\n const button = document.createElement('button')\n button.textContent = 'Test Connection'\n button.className = 'secondary'\n button.style.padding = '4px 9px'\n button.style.fontSize = '11px'\n\n const status = document.createElement('span')\n status.style.fontSize = '10px'\n status.style.opacity = '0.85'\n status.style.whiteSpace = 'normal'\n status.style.overflowWrap = 'anywhere'\n\n button.onclick = async () => {\n const startedAt = Date.now()\n button.disabled = true\n button.textContent = 'Testing...'\n status.textContent = 'Checking...'\n status.style.color = '#6b7280'\n\n try {\n const { testDeviceConnection } = await import('./api.js')\n const result = await testDeviceConnection({\n deviceId: String(device?.id || device?.deviceId || ''),\n connectionType: device?.connectionType,\n address: device?.address,\n })\n\n const measuredLatency = Number(result?.latencyMs) > 0\n ? Number(result.latencyMs)\n : Date.now() - startedAt\n\n if (result?.success) {\n const method = result?.method || 'Auto'\n status.textContent = `\u2713 ${method} \u00B7 ${measuredLatency}ms`\n status.style.color = '#16a34a'\n } else {\n const detail = result?.message ? ` \u00B7 ${result.message}` : ''\n status.textContent = `\u2717 Failed \u00B7 ${measuredLatency}ms${detail}`\n status.style.color = '#dc2626'\n }\n } catch (e) {\n status.textContent = `\u2717 Failed \u00B7 ${Date.now() - startedAt}ms`\n status.style.color = '#dc2626'\n } finally {\n button.disabled = false\n button.textContent = 'Test Connection'\n }\n }\n\n controls.appendChild(button)\n controls.appendChild(status)\n return controls\n}\n\nfunction formatLastSeen(value: any): string {\n if (!value) {\n return 'N/A'\n }\n try {\n const date = new Date(value)\n return Number.isNaN(date.getTime()) ? String(value) : date.toLocaleString()\n } catch (_e) {\n return String(value)\n }\n}\n\nexport function renderDeviceDetailsPanel(device: any): HTMLElement {\n const details = document.createElement('div')\n details.className = 'device-details-panel'\n details.style.borderTop = '1px solid #ddd'\n details.style.padding = '8px'\n details.style.borderRadius = '4px'\n details.style.fontSize = '12px'\n details.style.marginTop = '4px'\n\n // --- Battery history trending ---\n // Persist battery readings in localStorage per device\n const batteryHistoryKey = `batteryHistory:${device?.id || device?.deviceId}`\n let batteryHistory: Array<{ value: number, ts: number }> = []\n try {\n const raw = localStorage.getItem(batteryHistoryKey)\n if (raw) {\n batteryHistory = JSON.parse(raw)\n }\n } catch (e) {\n // Optionally log or handle error\n }\n const now = Date.now()\n if (typeof device?.battery === 'number') {\n // Only add if different from last or >1h since last\n const last = batteryHistory.at(-1)\n if (!last || last.value !== device.battery || now - last.ts > 60 * 60 * 1000) {\n batteryHistory.push({ value: device.battery, ts: now })\n // Keep only last 30 entries (about a month if daily)\n if (batteryHistory.length > 30) {\n batteryHistory = batteryHistory.slice(-30)\n }\n try {\n localStorage.setItem(batteryHistoryKey, JSON.stringify(batteryHistory))\n } catch (e) {\n // Optionally log or handle error\n }\n }\n }\n\n const rows: Array<{ label: string, value: string, copyable?: boolean }> = [\n { label: 'Name', value: String(device?.name || device?.configDeviceName || 'N/A') },\n { label: 'Device ID', value: String(device?.id || device?.deviceId || 'N/A'), copyable: !!(device?.id || device?.deviceId) },\n { label: 'MAC Address', value: String(device?.address || 'N/A'), copyable: !!device?.address },\n { label: 'Device Type', value: String(device?.type || device?.configDeviceType || 'N/A') },\n { label: 'Model', value: String(device?.model || 'N/A') },\n { label: 'Hub ID', value: String(device?.hubDeviceId || 'N/A') },\n { label: 'Battery', value: device?.battery !== undefined && device?.battery !== null ? `${device.battery}%` : 'N/A' },\n { label: 'Firmware', value: String(device?.version || device?.firmware || 'N/A') },\n { label: 'Cloud Service', value: device?.enabled === false ? 'Disabled' : 'Enabled' },\n { label: 'Last Seen', value: formatLastSeen(device?.lastSeen || device?.lastseen || device?.updatedAt) },\n ]\n\n for (const row of rows) {\n const line = document.createElement('div')\n line.style.display = 'flex'\n line.style.alignItems = 'center'\n line.style.justifyContent = 'space-between'\n line.style.gap = '8px'\n line.style.padding = '2px 0'\n\n const label = document.createElement('span')\n label.style.fontWeight = '600'\n label.style.minWidth = '110px'\n label.textContent = `${row.label}:`\n\n const valueWrap = document.createElement('span')\n valueWrap.style.display = 'inline-flex'\n valueWrap.style.alignItems = 'center'\n valueWrap.style.gap = '6px'\n valueWrap.style.flex = '1'\n valueWrap.style.justifyContent = 'flex-end'\n valueWrap.style.minWidth = '0'\n\n const value = document.createElement('span')\n value.style.fontFamily = 'monospace'\n value.style.fontSize = '11px'\n value.style.opacity = '0.9'\n value.style.whiteSpace = 'normal'\n value.style.overflowWrap = 'anywhere'\n value.style.wordBreak = 'break-word'\n value.style.textAlign = 'right'\n value.textContent = row.value\n\n valueWrap.appendChild(value)\n\n if (row.copyable && row.value && row.value !== 'N/A') {\n const copyBtn = document.createElement('button')\n copyBtn.textContent = '\uD83D\uDCCB'\n copyBtn.title = `Copy ${row.label}`\n copyBtn.style.padding = '2px 6px'\n copyBtn.style.fontSize = '10px'\n copyBtn.style.lineHeight = '1'\n copyBtn.style.background = '#e5e7eb'\n copyBtn.style.color = '#111827'\n\n copyBtn.onclick = async () => {\n try {\n await navigator.clipboard.writeText(row.value)\n copyBtn.textContent = '\u2713'\n setTimeout(() => {\n copyBtn.textContent = '\uD83D\uDCCB'\n }, 1200)\n } catch (_e) {\n copyBtn.textContent = '!'\n setTimeout(() => {\n copyBtn.textContent = '\uD83D\uDCCB'\n }, 1200)\n }\n }\n\n valueWrap.appendChild(copyBtn)\n }\n\n line.appendChild(label)\n line.appendChild(valueWrap)\n details.appendChild(line)\n\n // If this is the Battery row, add a sparkline chart below\n if (row.label === 'Battery' && Array.isArray(batteryHistory) && batteryHistory.length > 1) {\n const chart = document.createElement('div')\n chart.style.margin = '2px 0 8px 0'\n chart.style.width = '100%'\n chart.style.height = '28px'\n chart.style.display = 'flex'\n // SVG sparkline\n const w = 120\n const h = 24\n const pad = 2\n const min = Math.min(...batteryHistory.map(b => b.value), 100)\n const max = Math.max(...batteryHistory.map(b => b.value), 0)\n const range = max - min || 1\n const points = batteryHistory.map((b, i) => {\n const x = pad + (i * (w - 2 * pad)) / (batteryHistory.length - 1)\n const y = pad + (h - 2 * pad) * (1 - (b.value - min) / range)\n return `${x},${y}`\n }).join(' ')\n const svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg')\n svg.setAttribute('width', String(w))\n svg.setAttribute('height', String(h))\n svg.setAttribute('viewBox', `0 0 ${w} ${h}`)\n svg.style.display = 'block'\n svg.style.background = '#f3f4f6'\n svg.style.borderRadius = '3px'\n svg.style.marginTop = '2px'\n svg.style.boxShadow = '0 1px 2px #0001'\n // Polyline for trend\n const polyline = document.createElementNS('http://www.w3.org/2000/svg', 'polyline')\n polyline.setAttribute('points', points)\n polyline.setAttribute('fill', 'none')\n polyline.setAttribute('stroke', '#2563eb')\n polyline.setAttribute('stroke-width', '2')\n svg.appendChild(polyline)\n // Dots for each point\n batteryHistory.forEach((b, i) => {\n const x = pad + (i * (w - 2 * pad)) / (batteryHistory.length - 1)\n const y = pad + (h - 2 * pad) * (1 - (b.value - min) / range)\n const circle = document.createElementNS('http://www.w3.org/2000/svg', 'circle')\n circle.setAttribute('cx', String(x))\n circle.setAttribute('cy', String(y))\n circle.setAttribute('r', '2.5')\n circle.setAttribute('fill', '#2563eb')\n svg.appendChild(circle)\n })\n // Min/max labels\n const minLabel = document.createElementNS('http://www.w3.org/2000/svg', 'text')\n minLabel.setAttribute('x', '2')\n minLabel.setAttribute('y', String(h - 2))\n minLabel.setAttribute('font-size', '9')\n minLabel.setAttribute('fill', '#888')\n minLabel.textContent = `${min}%`\n svg.appendChild(minLabel)\n const maxLabel = document.createElementNS('http://www.w3.org/2000/svg', 'text')\n maxLabel.setAttribute('x', String(w - 18))\n maxLabel.setAttribute('y', '10')\n maxLabel.setAttribute('font-size', '9')\n maxLabel.setAttribute('fill', '#888')\n maxLabel.textContent = `${max}%`\n svg.appendChild(maxLabel)\n chart.appendChild(svg)\n details.appendChild(chart)\n }\n }\n\n // --- Expose advanced/extra features dynamically ---\n const featureKeys = [\n 'airQuality',\n 'pm25',\n 'pm10',\n 'voc',\n 'co2',\n 'humidity',\n 'temperature',\n 'preset',\n 'mode',\n 'presetMode',\n 'direction',\n 'calibration',\n 'multiCommand',\n 'extendedInfo',\n 'segmentedControl',\n 'features',\n 'capabilities',\n 'state',\n ]\n const shown = new Set(rows.map(r => r.label.toLowerCase().replace(SPACES_REGEX, '')))\n for (const key of featureKeys) {\n if (device && device[key] !== undefined && !shown.has(key.toLowerCase())) {\n const line = document.createElement('div')\n line.style.display = 'flex'\n line.style.alignItems = 'center'\n line.style.justifyContent = 'space-between'\n line.style.gap = '8px'\n line.style.padding = '2px 0'\n\n const label = document.createElement('span')\n label.style.fontWeight = '600'\n label.style.minWidth = '110px'\n label.textContent = `${key.replace(CAMELCASE_REGEX, ' $1').replace(FIRST_CHAR_REGEX, s => s.toUpperCase())}:`\n\n const value = document.createElement('span')\n value.style.fontFamily = 'monospace'\n value.style.fontSize = '11px'\n value.style.opacity = '0.9'\n value.style.whiteSpace = 'normal'\n value.style.overflowWrap = 'anywhere'\n value.style.wordBreak = 'break-word'\n value.style.textAlign = 'right'\n value.textContent = typeof device[key] === 'object' ? JSON.stringify(device[key]) : String(device[key])\n\n line.appendChild(label)\n line.appendChild(value)\n details.appendChild(line)\n }\n }\n\n return details\n}\n\nexport async function renderDiscoveredDevices(\n devices: any[],\n options: {\n configuredIds?: Set<string>\n selectedIds?: Set<string>\n onToggleSelect?: (device: any, selected: boolean) => void\n } = {},\n): Promise<HTMLElement> {\n const ul = document.createElement('ul')\n ul.className = 'device-grid'\n ul.style.maxHeight = '400px'\n ul.style.overflowY = 'auto'\n ul.style.marginTop = '12px'\n ul.style.padding = '0'\n ul.style.listStyle = 'none'\n\n const { addDeviceToConfig } = await import('./discovery.js')\n const { loadConfiguredDevices } = await import('./devices.js')\n const configuredIds = options.configuredIds ?? new Set<string>()\n const selectedIds = options.selectedIds ?? new Set<string>()\n const onToggleSelect = options.onToggleSelect\n\n for (const d of devices) {\n // Defensive check: warn if device is missing id, name, or type\n if (!d || (!d.id && !d.deviceId) || (!d.name && !d.type)) {\n console.warn('[SwitchBot][Discovery][renderDiscoveredDevices] Device missing required fields:', d)\n }\n const deviceId = normalizeId(d.id)\n const alreadyAdded = configuredIds.has(deviceId)\n\n const li = document.createElement('li')\n li.className = 'device-item'\n li.style.display = 'flex'\n li.style.flexDirection = 'column'\n li.style.alignItems = 'stretch'\n li.style.justifyContent = 'flex-start'\n li.style.padding = '5px 8px'\n li.style.marginBottom = '0'\n li.style.borderRadius = '5px'\n li.style.transition = 'all 0.2s ease'\n\n const info = document.createElement('div')\n info.style.flex = '1 1 auto'\n info.style.width = '100%'\n info.style.minWidth = '0'\n\n const nameContainer = document.createElement('div')\n nameContainer.style.display = 'flex'\n nameContainer.style.alignItems = 'center'\n nameContainer.style.marginBottom = '0'\n nameContainer.style.flexWrap = 'wrap'\n nameContainer.style.gap = '4px'\n\n const name = document.createElement('div')\n name.style.fontWeight = '500'\n name.style.fontSize = '13px'\n name.textContent = d.name || d.id\n\n const selectCheckbox = document.createElement('input')\n selectCheckbox.type = 'checkbox'\n selectCheckbox.style.width = 'auto'\n selectCheckbox.style.margin = '0 2px 0 0'\n selectCheckbox.checked = selectedIds.has(deviceId)\n if (alreadyAdded) {\n selectCheckbox.disabled = true\n selectCheckbox.title = 'Already configured'\n }\n selectCheckbox.onchange = () => {\n onToggleSelect?.(d, selectCheckbox.checked)\n // Notify listeners (e.g., batch buttons) of selection change\n window.dispatchEvent(new CustomEvent('discovery-selection-changed'))\n }\n\n nameContainer.appendChild(selectCheckbox)\n\n nameContainer.appendChild(name)\n\n // Show firmware update available indicator if present\n if (d.firmwareUpdateAvailable) {\n const fwBadge = document.createElement('span')\n fwBadge.textContent = 'Update Available'\n fwBadge.style.cssText = 'background: #fb923c; color: #111; padding: 1px 7px; border-radius: 3px; font-size: 10px; font-weight: 600; margin-left: 6px;'\n fwBadge.title = 'A firmware update is available for this device.'\n nameContainer.appendChild(fwBadge)\n }\n\n // Show offline/unreachable indicator if device is offline\n let offline = false\n const lastSeen = d.lastSeen || d.lastseen || d.updatedAt\n if (typeof d.offline === 'boolean') {\n offline = d.offline\n } else if (lastSeen) {\n try {\n const last = new Date(lastSeen).getTime()\n if (!Number.isNaN(last)) {\n if (Date.now() - last > 1000 * 60 * 60) { // 1 hour\n offline = true\n }\n }\n } catch {}\n }\n if (offline) {\n const offlineBadge = document.createElement('span')\n offlineBadge.textContent = 'Offline'\n offlineBadge.style.cssText = 'background: #dc2626; color: white; padding: 1px 7px; border-radius: 3px; font-size: 10px; font-weight: 600; margin-left: 6px;'\n offlineBadge.title = 'Device is offline or unreachable.'\n nameContainer.appendChild(offlineBadge)\n }\n\n const expandedDetails = document.createElement('div')\n expandedDetails.style.display = 'none'\n expandedDetails.appendChild(renderDeviceDetailsPanel(d))\n\n const expandBtn = document.createElement('button')\n expandBtn.textContent = '\u25BE'\n expandBtn.title = 'Show details'\n expandBtn.style.padding = '2px 6px'\n expandBtn.style.fontSize = '11px'\n expandBtn.style.marginLeft = '4px'\n expandBtn.style.background = '#e5e7eb'\n expandBtn.style.color = '#111827'\n expandBtn.style.transition = 'transform 0.2s ease'\n expandBtn.onclick = () => {\n const isHidden = expandedDetails.style.display === 'none'\n expandedDetails.style.display = isHidden ? 'block' : 'none'\n expandBtn.style.transform = isHidden ? 'rotate(180deg)' : 'rotate(0deg)'\n }\n nameContainer.appendChild(expandBtn)\n\n const duplicateBadge = document.createElement('span')\n duplicateBadge.textContent = alreadyAdded ? '\u2713 Already Added' : '\u2795 New Device'\n duplicateBadge.style.cssText = alreadyAdded\n ? 'background: #16a34a; color: white; padding: 1px 5px; border-radius: 3px; font-size: 9px; font-weight: 600;'\n : 'background: #2563eb; color: white; padding: 1px 5px; border-radius: 3px; font-size: 9px; font-weight: 600;'\n nameContainer.appendChild(duplicateBadge)\n\n // Add connection type badge\n if (d.connectionType) {\n const badge = renderConnectionBadge(d.connectionType)\n if (badge) {\n nameContainer.appendChild(badge)\n }\n }\n\n // Add IR badge if it's an IR device\n if (d.isIR) {\n nameContainer.appendChild(renderIRBadge())\n }\n\n // Add signal strength visualization (only for BLE/wireless devices)\n if (d.rssi !== undefined && d.rssi !== null && d.rssi !== 0) {\n nameContainer.appendChild(renderSignalBars(d.rssi))\n nameContainer.appendChild(renderSignalQualityBadge(d.rssi))\n }\n\n // Add battery warning indicator if battery < 20%\n if (typeof d.battery === 'number' && d.battery < 20) {\n const batteryWarn = document.createElement('span')\n batteryWarn.textContent = `\u26A0\uFE0F ${d.battery}%`\n batteryWarn.style.cssText\n = d.battery < 10\n ? 'background: #dc2626; color: white; padding: 1px 6px; border-radius: 3px; font-size: 10px; font-weight: 600; margin-left: 6px;'\n : 'background: #fbbf24; color: #111; padding: 1px 6px; border-radius: 3px; font-size: 10px; font-weight: 600; margin-left: 6px;'\n batteryWarn.title = d.battery < 10 ? 'Battery critically low' : 'Battery low'\n nameContainer.appendChild(batteryWarn)\n }\n\n // Defensive check: warn if device is missing id, name, or type (for details panel)\n if (!d || (!d.id && !d.deviceId) || (!d.name && !d.type)) {\n console.warn('[SwitchBot][Discovery][renderDeviceDetailsPanel] Device missing required fields:', d)\n }\n const details = document.createElement('div')\n details.style.fontSize = '10px'\n details.style.opacity = '0.7'\n details.style.marginTop = '0'\n details.style.fontFamily = 'monospace'\n details.style.whiteSpace = 'normal'\n details.style.overflowWrap = 'anywhere'\n details.style.wordBreak = 'break-word'\n\n let detailsText = `ID: ${d.id} | Type: ${d.type} | Model: ${d.model || 'N/A'}`\n if (d.hubDeviceId) {\n detailsText += ` | Hub: ${d.hubDeviceId}`\n }\n if (d.address) {\n detailsText += ` | MAC: ${d.address}`\n }\n details.textContent = detailsText\n\n info.appendChild(nameContainer)\n info.appendChild(details)\n info.appendChild(expandedDetails)\n\n const addBtn = document.createElement('button')\n addBtn.textContent = alreadyAdded ? 'Already Added' : 'Add to Config'\n addBtn.style.marginLeft = '0'\n addBtn.style.marginTop = '2px'\n addBtn.style.padding = '4px 9px'\n addBtn.style.fontSize = '11px'\n addBtn.style.whiteSpace = 'nowrap'\n addBtn.style.flexShrink = '0'\n addBtn.disabled = alreadyAdded\n if (alreadyAdded) {\n addBtn.style.opacity = '0.65'\n addBtn.style.cursor = 'not-allowed'\n addBtn.style.background = '#6b7280'\n }\n addBtn.onclick = async () => {\n if (alreadyAdded) {\n return\n }\n await addDeviceToConfig(d)\n }\n\n if (alreadyAdded) {\n const viewBtn = document.createElement('button')\n viewBtn.textContent = 'View in Config'\n viewBtn.className = 'secondary'\n viewBtn.style.marginLeft = '0'\n viewBtn.style.padding = '4px 9px'\n viewBtn.style.fontSize = '11px'\n viewBtn.onclick = async () => {\n await loadConfiguredDevices()\n scrollToConfiguredDevice(d.id)\n }\n li.appendChild(info)\n const actions = document.createElement('div')\n actions.className = 'device-actions'\n actions.style.display = 'flex'\n actions.style.alignItems = 'center'\n actions.style.flexWrap = 'wrap'\n actions.style.justifyContent = 'flex-start'\n actions.style.marginLeft = '0'\n actions.style.width = '100%'\n actions.style.marginTop = '2px'\n actions.style.gap = '5px'\n actions.appendChild(viewBtn)\n actions.appendChild(addBtn)\n actions.appendChild(createConnectionTestControls(d))\n li.appendChild(actions)\n ul.appendChild(li)\n continue\n }\n\n const actions = document.createElement('div')\n actions.className = 'device-actions'\n actions.style.display = 'flex'\n actions.style.flexWrap = 'wrap'\n actions.style.justifyContent = 'flex-start'\n actions.style.marginLeft = '0'\n actions.style.width = '100%'\n actions.style.marginTop = '2px'\n actions.style.gap = '5px'\n actions.appendChild(addBtn)\n actions.appendChild(createConnectionTestControls(d))\n\n li.appendChild(info)\n li.appendChild(actions)\n ul.appendChild(li)\n }\n\n return ul\n}\n\n/**\n * Filter devices by connection type and search query\n * @param devices Discovered devices array\n * @param connectionType Filter: 'all' | 'ble' | 'api' | 'both' | 'ir'\n * @param searchQuery Search term to match against name/id/type\n * @returns Filtered devices array\n */\nexport function filterDevices(\n devices: any[],\n connectionType: 'all' | 'ble' | 'api' | 'both' | 'ir' = 'all',\n searchQuery = '',\n): any[] {\n let filtered = [...devices]\n\n // Filter by connection type\n if (connectionType !== 'all') {\n filtered = filtered.filter((d) => {\n if (connectionType === 'ir') {\n return d.isIR === true\n }\n if (connectionType === 'ble') {\n return d.connectionType === 'BLE' || d.connectionType?.includes('BLE')\n }\n if (connectionType === 'api') {\n return d.connectionType === 'OpenAPI' || d.connectionType === 'API' || d.connectionType?.includes('API')\n }\n if (connectionType === 'both') {\n return d.connectionType === 'Both' || d.connectionType?.includes('Both')\n }\n return true\n })\n }\n\n // Filter by search query\n if (searchQuery.trim()) {\n const query = searchQuery.toLowerCase()\n filtered = filtered.filter((d) => {\n const name = (d.name || '').toLowerCase()\n const id = (d.id || '').toLowerCase()\n const type = (d.type || '').toLowerCase()\n const model = (d.model || '').toLowerCase()\n return name.includes(query) || id.includes(query) || type.includes(query) || model.includes(query)\n })\n }\n\n return filtered\n}\n\n/**\n * Sort devices by specified criteria\n * @param devices Devices array to sort\n * @param sortBy Sort criterion: 'name' | 'signal' | 'type' | 'connection'\n * @returns Sorted devices array\n */\nexport function sortDevices(\n devices: any[],\n sortBy: 'name' | 'signal' | 'type' | 'connection' = 'name',\n): any[] {\n const sorted = [...devices]\n\n switch (sortBy) {\n case 'signal': {\n // Sort by RSSI descending (strongest signal first)\n sorted.sort((a, b) => {\n const aRssi = a.rssi || 0\n const bRssi = b.rssi || 0\n return bRssi - aRssi // Descending order (higher is stronger)\n })\n break\n }\n case 'type': {\n // Sort by device type alphabetically\n sorted.sort((a, b) => {\n const aType = (a.type || '').localeCompare(b.type || '')\n return aType\n })\n break\n }\n case 'connection': {\n // Sort by connection type: Both > BLE > OpenAPI > Others\n const connectionOrder: Record<string, number> = {\n Both: 0,\n BLE: 1,\n OpenAPI: 2,\n API: 2,\n Unknown: 3,\n }\n sorted.sort((a, b) => {\n const aOrder = connectionOrder[a.connectionType || 'Unknown'] ?? 3\n const bOrder = connectionOrder[b.connectionType || 'Unknown'] ?? 3\n return aOrder - bOrder\n })\n break\n }\n case 'name':\n default: {\n // Sort by name alphabetically\n sorted.sort((a, b) => (a.name || a.id || '').localeCompare(b.name || b.id || ''))\n break\n }\n }\n\n return sorted\n}\n\n/**\n * Get filter/sort preferences from localStorage\n * @returns Object with current filter and sort preferences\n */\nexport function getDiscoveryPreferences(): {\n connectionType: 'all' | 'ble' | 'api' | 'both' | 'ir'\n sortBy: 'name' | 'signal' | 'type' | 'connection'\n searchQuery: string\n} {\n try {\n const stored = localStorage.getItem('discoveryPreferences')\n if (stored) {\n return JSON.parse(stored)\n }\n } catch (_e) {\n // Ignore parse errors\n }\n\n return {\n connectionType: 'all',\n sortBy: 'name',\n searchQuery: '',\n }\n}\n\n/**\n * Save filter/sort preferences to localStorage\n * @param preferences Preferences object to save\n * @param preferences.connectionType Connection type filter\n * @param preferences.sortBy Sort criterion\n * @param preferences.searchQuery Search query string\n */\nexport function setDiscoveryPreferences(preferences: {\n connectionType: 'all' | 'ble' | 'api' | 'both' | 'ir'\n sortBy: 'name' | 'signal' | 'type' | 'connection'\n searchQuery: string\n}): void {\n try {\n localStorage.setItem('discoveryPreferences', JSON.stringify(preferences))\n } catch (_e) {\n // Ignore storage errors\n }\n}\n\nexport function renderDeviceList(list: any[]): void {\n const ul = document.getElementById('devices')\n const status = document.getElementById('status')\n const removeAllContainer = document.getElementById('removeAllContainer')\n\n if (!ul || !status) {\n return\n }\n\n if (!list.length) {\n status.textContent = 'No devices found in config.'\n ul.innerHTML = ''\n // Hide remove all button when no devices\n if (removeAllContainer) {\n removeAllContainer.style.display = 'none'\n }\n return\n }\n\n status.textContent = `Found ${list.length} device(s)`\n ul.classList.add('device-grid')\n ul.innerHTML = ''\n\n // Show remove all button when devices exist\n if (removeAllContainer) {\n removeAllContainer.style.display = 'block'\n }\n\n for (const d of list) {\n const li = document.createElement('li')\n li.className = 'device-item'\n li.setAttribute('data-device-id', normalizeId(d.id))\n li.style.display = 'flex'\n li.style.flexDirection = 'column'\n li.style.alignItems = 'stretch'\n li.style.marginBottom = '0'\n\n const info = document.createElement('div')\n info.style.flex = '1 1 auto'\n info.style.width = '100%'\n info.style.minWidth = '0'\n\n const nameContainer = document.createElement('div')\n nameContainer.style.display = 'flex'\n nameContainer.style.flexDirection = 'column'\n nameContainer.style.alignItems = 'flex-start'\n nameContainer.style.gap = '0'\n\n const name = document.createElement('div')\n name.style.fontWeight = '500'\n name.style.fontSize = '13px'\n name.textContent = d.name || d.id\n\n const expandedDetails = document.createElement('div')\n expandedDetails.style.display = 'none'\n expandedDetails.appendChild(renderDeviceDetailsPanel(d))\n\n const expandBtn = document.createElement('button')\n expandBtn.textContent = '\u25BE'\n expandBtn.title = 'Show details'\n expandBtn.style.padding = '2px 6px'\n expandBtn.style.fontSize = '11px'\n expandBtn.style.marginLeft = '4px'\n expandBtn.style.background = '#e5e7eb'\n expandBtn.style.color = '#111827'\n expandBtn.style.transition = 'transform 0.2s ease'\n expandBtn.onclick = () => {\n const isHidden = expandedDetails.style.display === 'none'\n expandedDetails.style.display = isHidden ? 'block' : 'none'\n expandBtn.style.transform = isHidden ? 'rotate(180deg)' : 'rotate(0deg)'\n }\n\n const code = document.createElement('code')\n code.textContent = d.id\n code.style.fontSize = '10px'\n code.style.opacity = '0.75'\n code.style.marginLeft = '0'\n code.style.whiteSpace = 'normal'\n code.style.overflowWrap = 'anywhere'\n code.style.wordBreak = 'break-word'\n code.style.maxWidth = '100%'\n\n const headerRow = document.createElement('div')\n headerRow.style.display = 'inline-flex'\n headerRow.style.alignItems = 'center'\n headerRow.style.gap = '4px'\n headerRow.appendChild(name)\n headerRow.appendChild(expandBtn)\n\n nameContainer.appendChild(headerRow)\n nameContainer.appendChild(code)\n\n // Add signal strength visualization if RSSI is available\n if (d.rssi !== undefined && d.rssi !== null && d.rssi !== 0) {\n nameContainer.appendChild(renderSignalBars(d.rssi))\n nameContainer.appendChild(renderSignalQualityBadge(d.rssi))\n }\n\n const meta = document.createElement('div')\n meta.style.opacity = '0.75'\n meta.style.marginTop = '0'\n meta.style.fontSize = '11px'\n\n const typeText = d.type ? `type: ${d.type}` : ''\n const connText = d.connectionPreference ? `conn: ${d.connectionPreference}` : ''\n const roomText = d.room ? `room: ${d.room}` : ''\n meta.textContent = [typeText, connText, roomText].filter(Boolean).join(' | ')\n\n info.appendChild(nameContainer)\n info.appendChild(meta)\n info.appendChild(expandedDetails)\n\n const buttons = document.createElement('div')\n buttons.className = 'device-actions'\n buttons.style.display = 'flex'\n buttons.style.flexWrap = 'wrap'\n buttons.style.justifyContent = 'flex-start'\n buttons.style.marginLeft = '0'\n buttons.style.width = '100%'\n buttons.style.marginTop = '2px'\n buttons.style.gap = '5px'\n\n const editBtn = document.createElement('button')\n editBtn.textContent = '\u270F\uFE0F Edit'\n editBtn.style.padding = '4px 9px'\n editBtn.style.fontSize = '11px'\n editBtn.onclick = async () => {\n const { editDevice } = await import('./modals.js')\n await editDevice(d)\n }\n\n const copyBtn = document.createElement('button')\n copyBtn.textContent = 'Copy ID'\n copyBtn.style.padding = '4px 9px'\n copyBtn.style.fontSize = '11px'\n copyBtn.addEventListener('click', async () => {\n try {\n await navigator.clipboard.writeText(d.id)\n copyBtn.textContent = 'Copied'\n copyBtn.classList.add('success')\n setTimeout(() => {\n copyBtn.textContent = 'Copy ID'\n copyBtn.classList.remove('success')\n }, 1200)\n } catch (e) {\n copyBtn.textContent = 'Failed'\n copyBtn.classList.add('error')\n setTimeout(() => {\n copyBtn.textContent = 'Copy ID'\n copyBtn.classList.remove('error')\n }, 1200)\n }\n })\n\n const deleteBtn = document.createElement('button')\n deleteBtn.textContent = '\uD83D\uDDD1\uFE0F Delete'\n deleteBtn.style.padding = '4px 9px'\n deleteBtn.style.fontSize = '11px'\n deleteBtn.style.background = '#ef4444'\n deleteBtn.onclick = async () => {\n const { deleteDeviceFromConfig } = await import('./devices-delete.js')\n await deleteDeviceFromConfig(d.id || d.deviceId, d.name || d.id || d.deviceId)\n }\n\n buttons.appendChild(editBtn)\n buttons.appendChild(copyBtn)\n buttons.appendChild(createConnectionTestControls(d))\n buttons.appendChild(deleteBtn)\n\n li.appendChild(info)\n li.appendChild(buttons)\n ul.appendChild(li)\n }\n\n // No return value needed for void function\n}\n", "import { addDevice, fetchDevices, syncParentPluginConfigFromDisk } from './api.js'\nimport { uiLog } from './logger.js'\nimport { hideBusyUi, showBusyUi } from './modal.js'\nimport { renderDeviceList } from './render.js'\nimport { toastError, toastInfo, toastSuccess, toastWarning } from './toast.js'\n\nexport async function addDeviceToConfig(device: any, options: { refresh?: boolean, showStatus?: boolean } = {}): Promise<{ added: boolean }> {\n const { refresh = true, showStatus = true } = options\n try {\n const { importDiscoveredDevice } = await import('./modals.js')\n\n const importValues: any = await importDiscoveredDevice(device)\n if (!importValues || typeof importValues !== 'object' || !importValues.configDeviceName || !importValues.configDeviceType) {\n return { added: false }\n }\n\n showBusyUi()\n uiLog.info('Adding device to config:', device)\n // Never allow 'undefined' as a device name\n let safeName = importValues.configDeviceName\n if (!safeName || safeName === 'undefined') {\n safeName = device.name || device.id\n uiLog.warn(`Device name was invalid (\"${importValues.configDeviceName}\"), using fallback: \"${safeName}\"`)\n }\n const resp = await addDevice(device.id, safeName, importValues.configDeviceType, {\n address: importValues.address,\n model: device.model,\n rssi: device.rssi,\n encryptionKey: importValues.encryptionKey,\n keyId: importValues.keyId,\n })\n uiLog.info('Add device response:', resp)\n\n const alreadyExists = !!resp?.alreadyExists\n const message = resp?.message\n || (alreadyExists\n ? `Device \"${importValues.configDeviceName}\" already in config`\n : `Device \"${importValues.configDeviceName}\" added successfully!`)\n\n if (alreadyExists) {\n toastInfo(message)\n } else {\n toastSuccess(message)\n }\n\n if (showStatus) {\n const status = document.getElementById('discoverStatus')\n if (status) {\n status.textContent = (alreadyExists ? '\u2022 ' : '\u2713 ') + message\n status.classList.remove('error')\n status.classList.add('success-msg')\n\n // Sync parent Homebridge form cache and auto-save to prevent cache overwrite\n if (!alreadyExists) {\n const synced = await syncParentPluginConfigFromDisk(true)\n status.textContent += synced\n ? ' - Config saved automatically.'\n : ' - Warning: config may not persist until you close/reopen settings.'\n\n if (synced) {\n toastSuccess('Configuration synced and saved automatically')\n } else {\n toastWarning('Configuration sync failed; close and reopen settings before Save')\n }\n }\n }\n }\n\n if (refresh) {\n // Force reload of config from disk after add\n await syncParentPluginConfigFromDisk(true)\n await loadConfiguredDevices()\n }\n\n return { added: !alreadyExists }\n } catch (e) {\n const msg = e instanceof Error ? e.message : String(e)\n uiLog.error('Add device error:', msg)\n const status = document.getElementById('discoverStatus')\n if (status) {\n status.textContent = `\u2717 Error: ${msg}`\n status.classList.add('error')\n }\n toastError(msg)\n return { added: false }\n } finally {\n hideBusyUi()\n }\n}\n\nexport async function initRemoveAllButton(): Promise<void> {\n const removeAllBtn = document.getElementById('removeAllBtn')\n if (!removeAllBtn) {\n return\n }\n\n removeAllBtn.addEventListener('click', async () => {\n const { deleteAllDevicesFromConfig } = await import('./devices-delete.js')\n await deleteAllDevicesFromConfig()\n })\n}\n\nexport async function loadConfiguredDevices(): Promise<void> {\n const list = await fetchDevices()\n renderDeviceList(list)\n}\n", "import { uiLog } from './logger.js'\nimport { hideBusyUi, showBusyUi } from './modal.js'\nimport { toastError, toastSuccess, toastWarning } from './toast.js'\n\nexport async function loadCredentialStatus(): Promise<void> {\n try {\n if (typeof homebridge.getPluginConfig !== 'function') {\n uiLog.error('Homebridge UI API not available')\n return\n }\n const configArr = await homebridge.getPluginConfig()\n const config = Array.isArray(configArr) && configArr.length > 0 ? configArr[0] : {}\n const token = config.openApiToken || ''\n const secret = config.openApiSecret || ''\n\n const tokenStatus = document.getElementById('tokenStatus')\n const secretStatus = document.getElementById('secretStatus')\n if (!tokenStatus || !secretStatus) {\n return\n }\n if (token) {\n tokenStatus.textContent = `\u2713 Configured (${token.length} characters)`\n tokenStatus.classList.add('ok')\n } else {\n tokenStatus.textContent = 'Not configured'\n tokenStatus.classList.remove('ok')\n }\n if (secret) {\n secretStatus.textContent = `\u2713 Configured (${secret.length} characters)`\n secretStatus.classList.add('ok')\n } else {\n secretStatus.textContent = 'Not configured'\n secretStatus.classList.remove('ok')\n }\n } catch (e) {\n uiLog.error('Error loading credentials:', e)\n }\n}\n\nexport async function saveCredentials(): Promise<void> {\n const token = (document.getElementById('token') as HTMLInputElement)?.value\n const secret = (document.getElementById('secret') as HTMLInputElement)?.value\n const saveStatus = document.getElementById('saveStatus')\n const saveBtn = document.getElementById('saveBtn') as HTMLButtonElement\n\n if (!saveStatus || !saveBtn) {\n return\n }\n\n if (!token || !secret) {\n saveStatus.textContent = 'Please enter both token and secret'\n saveStatus.classList.add('error')\n toastWarning('Please enter both token and secret')\n return\n }\n\n try {\n showBusyUi()\n saveBtn.disabled = true\n saveBtn.textContent = 'Saving...'\n uiLog.info('Saving credentials...')\n\n if (typeof homebridge.getPluginConfig !== 'function' || typeof homebridge.updatePluginConfig !== 'function') {\n throw new TypeError('Homebridge UI API not available')\n }\n const configArr = await homebridge.getPluginConfig()\n if (!Array.isArray(configArr) || configArr.length === 0) {\n throw new Error('No plugin config found')\n }\n const config = configArr[0]\n config.openApiToken = token\n config.openApiSecret = secret\n await homebridge.updatePluginConfig([config])\n if (typeof homebridge.savePluginConfig === 'function') {\n await homebridge.savePluginConfig()\n }\n\n saveStatus.textContent = `\u2713 Credentials saved successfully`\n saveStatus.classList.remove('error')\n saveStatus.classList.add('success-msg')\n toastSuccess('Credentials saved successfully')\n\n // Clear inputs after successful save\n ;(document.getElementById('token') as HTMLInputElement).value = ''\n ;(document.getElementById('secret') as HTMLInputElement).value = ''\n\n // Reload status to verify save\n setTimeout(loadCredentialStatus, 500)\n\n // Clear status message\n setTimeout(() => {\n saveStatus.textContent = ''\n saveStatus.classList.remove('success-msg')\n }, 3000)\n } catch (e) {\n uiLog.error('Save error:', e)\n saveStatus.textContent = `Error: ${e instanceof Error ? e.message : 'Failed to save'}`\n saveStatus.classList.add('error')\n toastError(e instanceof Error ? e.message : 'Failed to save credentials')\n } finally {\n hideBusyUi()\n saveBtn.disabled = false\n saveBtn.textContent = 'Save Credentials'\n }\n}\n", "import { loadCredentialStatus, saveCredentials } from './credentials.js'\nimport { initRemoveAllButton, loadConfiguredDevices } from './devices.js'\nimport { discoverDevices, initializeDiscoverySettings } from './discovery.js';\n\n(window as any).loadCredentialStatus = loadCredentialStatus;\n(window as any).saveCredentials = saveCredentials;\n(window as any).discoverDevices = discoverDevices\n\n// Initialize on page load\nasync function init(): Promise<void> {\n await loadCredentialStatus()\n await initializeDiscoverySettings()\n await loadConfiguredDevices()\n await initRemoveAllButton()\n}\n\n// Run initialization when DOM is ready\nif (document.readyState === 'loading') {\n document.addEventListener('DOMContentLoaded', init)\n} else {\n init()\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": ["saveCredentials", "discoverDevices", "selectedIds", "addDevice", "updateDevice", "syncParentPluginConfigFromDisk", "fetchDevices", "renderDeviceList", "normalizeId", "testDeviceConnection", "addDeviceToConfig", "loadConfiguredDevices", "actions", "editDevice", "deleteDeviceFromConfig", "addDeviceToConfig", "importDiscoveredDevice", "deleteAllDevicesFromConfig", "discoverDevices"]
|
|
7
|
+
}
|