@switchbot/homebridge-switchbot 5.0.0-beta.15 → 5.0.0-beta.150

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (610) hide show
  1. package/.changeset/config.json +14 -0
  2. package/.github/ISSUE_TEMPLATE/e2e-verification.md +36 -0
  3. package/.github/copilot-instructions.md +39 -0
  4. package/.github/workflows/ci.yml +32 -0
  5. package/.github/workflows/manual-e2e.yml +115 -0
  6. package/.github/workflows/release.yml +0 -4
  7. package/.husky/pre-push +15 -0
  8. package/CHANGELOG.md +35 -0
  9. package/E2E-VERIFICATION.md +121 -0
  10. package/MIGRATION.md +54 -0
  11. package/README.md +136 -4
  12. package/TODO.md +263 -0
  13. package/config.schema.json +284 -14787
  14. package/dist/SwitchBotHAPPlatform.d.ts +133 -0
  15. package/dist/SwitchBotHAPPlatform.d.ts.map +1 -0
  16. package/dist/SwitchBotHAPPlatform.js +555 -0
  17. package/dist/SwitchBotHAPPlatform.js.map +1 -0
  18. package/dist/SwitchBotMatterPlatform.d.ts +141 -0
  19. package/dist/SwitchBotMatterPlatform.d.ts.map +1 -0
  20. package/dist/SwitchBotMatterPlatform.js +536 -0
  21. package/dist/SwitchBotMatterPlatform.js.map +1 -0
  22. package/dist/device-types.d.ts +31 -0
  23. package/dist/device-types.d.ts.map +1 -0
  24. package/dist/device-types.js +246 -0
  25. package/dist/device-types.js.map +1 -0
  26. package/dist/deviceCommandMapper.d.ts +10 -0
  27. package/dist/deviceCommandMapper.d.ts.map +1 -0
  28. package/dist/deviceCommandMapper.js +319 -0
  29. package/dist/deviceCommandMapper.js.map +1 -0
  30. package/dist/deviceFactory.d.ts +14 -0
  31. package/dist/deviceFactory.d.ts.map +1 -0
  32. package/dist/deviceFactory.js +159 -0
  33. package/dist/deviceFactory.js.map +1 -0
  34. package/dist/devices/deviceBase.d.ts +50 -0
  35. package/dist/devices/deviceBase.d.ts.map +1 -0
  36. package/dist/devices/deviceBase.js +119 -0
  37. package/dist/devices/deviceBase.js.map +1 -0
  38. package/dist/devices/genericDevice.d.ts +320 -0
  39. package/dist/devices/genericDevice.d.ts.map +1 -0
  40. package/dist/devices/genericDevice.js +1363 -0
  41. package/dist/devices/genericDevice.js.map +1 -0
  42. package/dist/errors.d.ts +38 -0
  43. package/dist/errors.d.ts.map +1 -0
  44. package/dist/errors.js +32 -0
  45. package/dist/errors.js.map +1 -0
  46. package/dist/homebridge-ui/endpoints/config.d.ts +3 -0
  47. package/dist/homebridge-ui/endpoints/config.d.ts.map +1 -0
  48. package/dist/homebridge-ui/endpoints/config.js +90 -0
  49. package/dist/homebridge-ui/endpoints/config.js.map +1 -0
  50. package/dist/homebridge-ui/endpoints/devices.d.ts +6 -0
  51. package/dist/homebridge-ui/endpoints/devices.d.ts.map +1 -0
  52. package/dist/homebridge-ui/endpoints/devices.js +81 -0
  53. package/dist/homebridge-ui/endpoints/devices.js.map +1 -0
  54. package/dist/homebridge-ui/endpoints/discovery.d.ts +7 -0
  55. package/dist/homebridge-ui/endpoints/discovery.d.ts.map +1 -0
  56. package/dist/homebridge-ui/endpoints/discovery.js +212 -0
  57. package/dist/homebridge-ui/endpoints/discovery.js.map +1 -0
  58. package/dist/homebridge-ui/public/css/styles.css +472 -0
  59. package/dist/homebridge-ui/public/index.html +210 -244
  60. package/dist/homebridge-ui/public/js/advanced-settings.d.ts +3 -0
  61. package/dist/homebridge-ui/public/js/advanced-settings.d.ts.map +1 -0
  62. package/dist/homebridge-ui/public/js/advanced-settings.js +95 -0
  63. package/dist/homebridge-ui/public/js/advanced-settings.js.map +1 -0
  64. package/dist/homebridge-ui/public/js/advanced-settings.ts +94 -0
  65. package/dist/homebridge-ui/public/js/api.d.ts +66 -0
  66. package/dist/homebridge-ui/public/js/api.d.ts.map +1 -0
  67. package/dist/homebridge-ui/public/js/api.js +281 -0
  68. package/dist/homebridge-ui/public/js/api.js.map +1 -0
  69. package/dist/homebridge-ui/public/js/api.ts +342 -0
  70. package/dist/homebridge-ui/public/js/app.d.ts +2 -0
  71. package/dist/homebridge-ui/public/js/app.d.ts.map +1 -0
  72. package/dist/homebridge-ui/public/js/app.js +3863 -0
  73. package/dist/homebridge-ui/public/js/app.js.map +7 -0
  74. package/dist/homebridge-ui/public/js/app.ts +22 -0
  75. package/dist/homebridge-ui/public/js/constants.d.ts +2 -0
  76. package/dist/homebridge-ui/public/js/constants.d.ts.map +1 -0
  77. package/dist/homebridge-ui/public/js/constants.js +2 -0
  78. package/dist/homebridge-ui/public/js/constants.js.map +1 -0
  79. package/dist/homebridge-ui/public/js/constants.ts +1 -0
  80. package/dist/homebridge-ui/public/js/credentials.d.ts +3 -0
  81. package/dist/homebridge-ui/public/js/credentials.d.ts.map +1 -0
  82. package/dist/homebridge-ui/public/js/credentials.js +99 -0
  83. package/dist/homebridge-ui/public/js/credentials.js.map +1 -0
  84. package/dist/homebridge-ui/public/js/credentials.ts +105 -0
  85. package/dist/homebridge-ui/public/js/devices-delete.d.ts +3 -0
  86. package/dist/homebridge-ui/public/js/devices-delete.d.ts.map +1 -0
  87. package/dist/homebridge-ui/public/js/devices-delete.js +199 -0
  88. package/dist/homebridge-ui/public/js/devices-delete.js.map +1 -0
  89. package/dist/homebridge-ui/public/js/devices-delete.ts +227 -0
  90. package/dist/homebridge-ui/public/js/devices.d.ts +9 -0
  91. package/dist/homebridge-ui/public/js/devices.d.ts.map +1 -0
  92. package/dist/homebridge-ui/public/js/devices.js +96 -0
  93. package/dist/homebridge-ui/public/js/devices.js.map +1 -0
  94. package/dist/homebridge-ui/public/js/devices.ts +105 -0
  95. package/dist/homebridge-ui/public/js/discovery.d.ts +4 -0
  96. package/dist/homebridge-ui/public/js/discovery.d.ts.map +1 -0
  97. package/dist/homebridge-ui/public/js/discovery.js +1374 -0
  98. package/dist/homebridge-ui/public/js/discovery.js.map +1 -0
  99. package/dist/homebridge-ui/public/js/discovery.ts +1498 -0
  100. package/dist/homebridge-ui/public/js/logger.d.ts +7 -0
  101. package/dist/homebridge-ui/public/js/logger.d.ts.map +1 -0
  102. package/dist/homebridge-ui/public/js/logger.js +17 -0
  103. package/dist/homebridge-ui/public/js/logger.js.map +1 -0
  104. package/dist/homebridge-ui/public/js/logger.ts +17 -0
  105. package/dist/homebridge-ui/public/js/modal.d.ts +5 -0
  106. package/dist/homebridge-ui/public/js/modal.d.ts.map +1 -0
  107. package/dist/homebridge-ui/public/js/modal.js +26 -0
  108. package/dist/homebridge-ui/public/js/modal.js.map +1 -0
  109. package/dist/homebridge-ui/public/js/modal.ts +28 -0
  110. package/dist/homebridge-ui/public/js/modals.d.ts +15 -0
  111. package/dist/homebridge-ui/public/js/modals.d.ts.map +1 -0
  112. package/dist/homebridge-ui/public/js/modals.js +673 -0
  113. package/dist/homebridge-ui/public/js/modals.js.map +1 -0
  114. package/dist/homebridge-ui/public/js/modals.ts +761 -0
  115. package/dist/homebridge-ui/public/js/render.d.ts +71 -0
  116. package/dist/homebridge-ui/public/js/render.d.ts.map +1 -0
  117. package/dist/homebridge-ui/public/js/render.js +953 -0
  118. package/dist/homebridge-ui/public/js/render.js.map +1 -0
  119. package/dist/homebridge-ui/public/js/render.ts +1077 -0
  120. package/dist/homebridge-ui/public/js/toast.d.ts +6 -0
  121. package/dist/homebridge-ui/public/js/toast.d.ts.map +1 -0
  122. package/dist/homebridge-ui/public/js/toast.js +29 -0
  123. package/dist/homebridge-ui/public/js/toast.js.map +1 -0
  124. package/dist/homebridge-ui/public/js/toast.ts +37 -0
  125. package/dist/homebridge-ui/public/js/types.d.ts +23 -0
  126. package/dist/homebridge-ui/public/js/types.d.ts.map +1 -0
  127. package/dist/homebridge-ui/public/js/types.js +2 -0
  128. package/dist/homebridge-ui/public/js/types.js.map +1 -0
  129. package/dist/homebridge-ui/public/js/types.ts +26 -0
  130. package/dist/homebridge-ui/server.js +9 -41
  131. package/dist/homebridge-ui/server.js.map +1 -1
  132. package/dist/homebridge-ui/utils/config-parser.d.ts +35 -0
  133. package/dist/homebridge-ui/utils/config-parser.d.ts.map +1 -0
  134. package/dist/homebridge-ui/utils/config-parser.js +87 -0
  135. package/dist/homebridge-ui/utils/config-parser.js.map +1 -0
  136. package/dist/homebridge-ui/utils/device-migration.d.ts +35 -0
  137. package/dist/homebridge-ui/utils/device-migration.d.ts.map +1 -0
  138. package/dist/homebridge-ui/utils/device-migration.js +111 -0
  139. package/dist/homebridge-ui/utils/device-migration.js.map +1 -0
  140. package/dist/homebridge-ui/utils/logger.d.ts +7 -0
  141. package/dist/homebridge-ui/utils/logger.d.ts.map +1 -0
  142. package/dist/homebridge-ui/utils/logger.js +17 -0
  143. package/dist/homebridge-ui/utils/logger.js.map +1 -0
  144. package/dist/index.d.ts +10 -0
  145. package/dist/index.d.ts.map +1 -1
  146. package/dist/index.js +12 -4
  147. package/dist/index.js.map +1 -1
  148. package/dist/settings.d.ts +11 -249
  149. package/dist/settings.d.ts.map +1 -1
  150. package/dist/settings.js +6 -30
  151. package/dist/settings.js.map +1 -1
  152. package/dist/switchbotClient.d.ts +28 -0
  153. package/dist/switchbotClient.d.ts.map +1 -0
  154. package/dist/switchbotClient.js +175 -0
  155. package/dist/switchbotClient.js.map +1 -0
  156. package/dist/utils.d.ts +92 -115
  157. package/dist/utils.d.ts.map +1 -1
  158. package/dist/utils.js +1117 -902
  159. package/dist/utils.js.map +1 -1
  160. package/docs/assets/highlight.css +28 -0
  161. package/docs/assets/icons.js +1 -1
  162. package/docs/assets/icons.svg +1 -1
  163. package/docs/assets/main.js +2 -2
  164. package/docs/assets/style.css +3 -3
  165. package/docs/index.html +148 -13
  166. package/docs/variables/default.html +3 -1
  167. package/eslint.config.js +7 -8
  168. package/nodemon.json +2 -2
  169. package/package.json +36 -32
  170. package/scripts/build-ui.js +37 -0
  171. package/scripts/e2e/README.md +25 -0
  172. package/scripts/e2e/curtain-e2e.sh +70 -0
  173. package/scripts/e2e/fan-e2e.sh +75 -0
  174. package/scripts/e2e/light-advanced-e2e.sh +97 -0
  175. package/scripts/e2e/light-e2e.sh +75 -0
  176. package/scripts/e2e/list-accessories.sh +19 -0
  177. package/scripts/e2e/lock-e2e.sh +65 -0
  178. package/scripts/free-dev-ports.mjs +105 -0
  179. package/scripts/generate-matter-maps.js +77 -0
  180. package/scripts/run-e2e-local.sh +14 -0
  181. package/scripts/sync-device-types.mjs +31 -0
  182. package/src/SwitchBotHAPPlatform.ts +558 -0
  183. package/src/SwitchBotMatterPlatform.ts +538 -0
  184. package/src/device-types.ts +261 -0
  185. package/src/deviceCommandMapper.ts +333 -0
  186. package/src/deviceFactory.ts +201 -0
  187. package/src/devices/deviceBase.ts +141 -0
  188. package/src/devices/genericDevice.ts +1337 -0
  189. package/src/errors.ts +35 -0
  190. package/src/homebridge-ui/endpoints/config.ts +110 -0
  191. package/src/homebridge-ui/endpoints/devices.ts +88 -0
  192. package/src/homebridge-ui/endpoints/discovery.ts +233 -0
  193. package/src/homebridge-ui/public/css/styles.css +472 -0
  194. package/src/homebridge-ui/public/index.html +210 -244
  195. package/src/homebridge-ui/public/js/advanced-settings.ts +94 -0
  196. package/src/homebridge-ui/public/js/api.ts +342 -0
  197. package/src/homebridge-ui/public/js/app.ts +22 -0
  198. package/src/homebridge-ui/public/js/constants.ts +1 -0
  199. package/src/homebridge-ui/public/js/credentials.ts +105 -0
  200. package/src/homebridge-ui/public/js/devices-delete.ts +227 -0
  201. package/src/homebridge-ui/public/js/devices.ts +105 -0
  202. package/src/homebridge-ui/public/js/discovery.ts +1498 -0
  203. package/src/homebridge-ui/public/js/logger.ts +17 -0
  204. package/src/homebridge-ui/public/js/modal.ts +28 -0
  205. package/src/homebridge-ui/public/js/modals.ts +761 -0
  206. package/src/homebridge-ui/public/js/render.ts +1077 -0
  207. package/src/homebridge-ui/public/js/toast.ts +37 -0
  208. package/src/homebridge-ui/public/js/types.ts +26 -0
  209. package/src/homebridge-ui/server.ts +9 -43
  210. package/src/homebridge-ui/utils/config-parser.ts +108 -0
  211. package/src/homebridge-ui/utils/device-migration.ts +144 -0
  212. package/src/homebridge-ui/utils/logger.ts +17 -0
  213. package/src/index.ts +12 -4
  214. package/src/settings.ts +14 -277
  215. package/src/switchbotClient.ts +181 -0
  216. package/src/utils.ts +1106 -900
  217. package/test/client/switchbot-client-debounce.spec.ts +35 -0
  218. package/test/client/switchbot-client-openapi.spec.ts +19 -0
  219. package/test/client/switchbotClient.spec.ts +23 -0
  220. package/test/device/device-mapping.spec.ts +23 -0
  221. package/test/device/deviceBase.spec.ts +26 -0
  222. package/test/device/deviceFactory-edge.spec.ts +15 -0
  223. package/test/device/deviceFactory.spec.ts +33 -0
  224. package/test/device/fan-swing.spec.ts +34 -0
  225. package/test/device/genericDevice-blepoll.spec.ts +47 -0
  226. package/test/device/irdevice.spec.ts +9 -0
  227. package/test/device/lock-users.spec.ts +35 -0
  228. package/test/device/matter-descriptors.spec.ts +22 -0
  229. package/test/device/matter-device-state.spec.ts +37 -0
  230. package/test/e2e/run-e2e.spec.ts +48 -0
  231. package/test/errors/errors.spec.ts +10 -0
  232. package/test/helpers/matter-harness.ts +64 -0
  233. package/test/homebridge-ui/server.spec.ts +9 -0
  234. package/test/platform/accessory-restore.spec.ts +37 -0
  235. package/test/platform/matter-childbridge.spec.ts +34 -0
  236. package/test/platform/matter-integration.spec.ts +33 -0
  237. package/test/platform/platform-edge.spec.ts +73 -0
  238. package/test/platform/platform.integration.spec.ts +34 -0
  239. package/test/utils/utils-extra.spec.ts +10 -0
  240. package/test/utils/utils.spec.ts +53 -0
  241. package/todo/TODO.md +80 -0
  242. package/vitest.config.ts +7 -0
  243. package/coverage/base.css +0 -224
  244. package/coverage/block-navigation.js +0 -87
  245. package/coverage/clover.xml +0 -15847
  246. package/coverage/coverage-final.json +0 -42
  247. package/coverage/docs/assets/dmt/dmt-component-data.js.html +0 -85
  248. package/coverage/docs/assets/dmt/dmt-components.js.html +0 -286
  249. package/coverage/docs/assets/dmt/index.html +0 -131
  250. package/coverage/docs/assets/hierarchy.js.html +0 -85
  251. package/coverage/docs/assets/icons.js.html +0 -136
  252. package/coverage/docs/assets/index.html +0 -146
  253. package/coverage/docs/assets/main.js.html +0 -265
  254. package/coverage/favicon.png +0 -0
  255. package/coverage/index.html +0 -191
  256. package/coverage/prettify.css +0 -1
  257. package/coverage/prettify.js +0 -2
  258. package/coverage/sort-arrow-sprite.png +0 -0
  259. package/coverage/sorter.js +0 -196
  260. package/coverage/src/device/blindtilt.ts.html +0 -3238
  261. package/coverage/src/device/bot.ts.html +0 -2803
  262. package/coverage/src/device/ceilinglight.ts.html +0 -2338
  263. package/coverage/src/device/colorbulb.ts.html +0 -2824
  264. package/coverage/src/device/contact.ts.html +0 -1465
  265. package/coverage/src/device/curtain.ts.html +0 -2869
  266. package/coverage/src/device/device.ts.html +0 -2500
  267. package/coverage/src/device/fan.ts.html +0 -2242
  268. package/coverage/src/device/hub.ts.html +0 -1408
  269. package/coverage/src/device/humidifier.ts.html +0 -2116
  270. package/coverage/src/device/index.html +0 -416
  271. package/coverage/src/device/iosensor.ts.html +0 -1375
  272. package/coverage/src/device/lightstrip.ts.html +0 -2617
  273. package/coverage/src/device/lock.ts.html +0 -1963
  274. package/coverage/src/device/meter.ts.html +0 -1372
  275. package/coverage/src/device/meterplus.ts.html +0 -1384
  276. package/coverage/src/device/meterpro.ts.html +0 -1618
  277. package/coverage/src/device/motion.ts.html +0 -1264
  278. package/coverage/src/device/plug.ts.html +0 -1372
  279. package/coverage/src/device/relayswitch.ts.html +0 -2284
  280. package/coverage/src/device/robotvacuumcleaner.ts.html +0 -1810
  281. package/coverage/src/device/waterdetector.ts.html +0 -1294
  282. package/coverage/src/homebridge-ui/index.html +0 -116
  283. package/coverage/src/homebridge-ui/server.ts.html +0 -229
  284. package/coverage/src/index.html +0 -161
  285. package/coverage/src/index.ts.html +0 -124
  286. package/coverage/src/irdevice/airconditioner.ts.html +0 -1687
  287. package/coverage/src/irdevice/airpurifier.ts.html +0 -844
  288. package/coverage/src/irdevice/camera.ts.html +0 -475
  289. package/coverage/src/irdevice/fan.ts.html +0 -766
  290. package/coverage/src/irdevice/index.html +0 -251
  291. package/coverage/src/irdevice/irdevice.ts.html +0 -1117
  292. package/coverage/src/irdevice/light.ts.html +0 -826
  293. package/coverage/src/irdevice/other.ts.html +0 -2458
  294. package/coverage/src/irdevice/tv.ts.html +0 -1222
  295. package/coverage/src/irdevice/vacuumcleaner.ts.html +0 -466
  296. package/coverage/src/irdevice/waterheater.ts.html +0 -469
  297. package/coverage/src/platform.ts.html +0 -8776
  298. package/coverage/src/settings.ts.html +0 -934
  299. package/coverage/src/utils.ts.html +0 -2092
  300. package/dist/baseMatterAccessory.test.d.ts +0 -2
  301. package/dist/baseMatterAccessory.test.d.ts.map +0 -1
  302. package/dist/baseMatterAccessory.test.js +0 -71
  303. package/dist/baseMatterAccessory.test.js.map +0 -1
  304. package/dist/devices-hap/airpurifier.d.ts +0 -54
  305. package/dist/devices-hap/airpurifier.d.ts.map +0 -1
  306. package/dist/devices-hap/airpurifier.js +0 -527
  307. package/dist/devices-hap/airpurifier.js.map +0 -1
  308. package/dist/devices-hap/blindtilt.d.ts +0 -90
  309. package/dist/devices-hap/blindtilt.d.ts.map +0 -1
  310. package/dist/devices-hap/blindtilt.js +0 -974
  311. package/dist/devices-hap/blindtilt.js.map +0 -1
  312. package/dist/devices-hap/bot.d.ts +0 -102
  313. package/dist/devices-hap/bot.d.ts.map +0 -1
  314. package/dist/devices-hap/bot.js +0 -811
  315. package/dist/devices-hap/bot.js.map +0 -1
  316. package/dist/devices-hap/ceilinglight.d.ts +0 -85
  317. package/dist/devices-hap/ceilinglight.d.ts.map +0 -1
  318. package/dist/devices-hap/ceilinglight.js +0 -701
  319. package/dist/devices-hap/ceilinglight.js.map +0 -1
  320. package/dist/devices-hap/colorbulb.d.ts +0 -88
  321. package/dist/devices-hap/colorbulb.d.ts.map +0 -1
  322. package/dist/devices-hap/colorbulb.js +0 -881
  323. package/dist/devices-hap/colorbulb.js.map +0 -1
  324. package/dist/devices-hap/contact.d.ts +0 -44
  325. package/dist/devices-hap/contact.d.ts.map +0 -1
  326. package/dist/devices-hap/contact.js +0 -409
  327. package/dist/devices-hap/contact.js.map +0 -1
  328. package/dist/devices-hap/curtain.d.ts +0 -73
  329. package/dist/devices-hap/curtain.d.ts.map +0 -1
  330. package/dist/devices-hap/curtain.js +0 -869
  331. package/dist/devices-hap/curtain.js.map +0 -1
  332. package/dist/devices-hap/device.d.ts +0 -98
  333. package/dist/devices-hap/device.d.ts.map +0 -1
  334. package/dist/devices-hap/device.js +0 -831
  335. package/dist/devices-hap/device.js.map +0 -1
  336. package/dist/devices-hap/fan.d.ts +0 -69
  337. package/dist/devices-hap/fan.d.ts.map +0 -1
  338. package/dist/devices-hap/fan.js +0 -649
  339. package/dist/devices-hap/fan.js.map +0 -1
  340. package/dist/devices-hap/hub.d.ts +0 -37
  341. package/dist/devices-hap/hub.d.ts.map +0 -1
  342. package/dist/devices-hap/hub.js +0 -392
  343. package/dist/devices-hap/hub.js.map +0 -1
  344. package/dist/devices-hap/humidifier.d.ts +0 -68
  345. package/dist/devices-hap/humidifier.d.ts.map +0 -1
  346. package/dist/devices-hap/humidifier.js +0 -628
  347. package/dist/devices-hap/humidifier.js.map +0 -1
  348. package/dist/devices-hap/iosensor.d.ts +0 -42
  349. package/dist/devices-hap/iosensor.d.ts.map +0 -1
  350. package/dist/devices-hap/iosensor.js +0 -382
  351. package/dist/devices-hap/iosensor.js.map +0 -1
  352. package/dist/devices-hap/lightstrip.d.ts +0 -79
  353. package/dist/devices-hap/lightstrip.d.ts.map +0 -1
  354. package/dist/devices-hap/lightstrip.js +0 -797
  355. package/dist/devices-hap/lightstrip.js.map +0 -1
  356. package/dist/devices-hap/lock.d.ts +0 -53
  357. package/dist/devices-hap/lock.d.ts.map +0 -1
  358. package/dist/devices-hap/lock.js +0 -561
  359. package/dist/devices-hap/lock.js.map +0 -1
  360. package/dist/devices-hap/meter.d.ts +0 -37
  361. package/dist/devices-hap/meter.d.ts.map +0 -1
  362. package/dist/devices-hap/meter.js +0 -379
  363. package/dist/devices-hap/meter.js.map +0 -1
  364. package/dist/devices-hap/meterplus.d.ts +0 -42
  365. package/dist/devices-hap/meterplus.d.ts.map +0 -1
  366. package/dist/devices-hap/meterplus.js +0 -384
  367. package/dist/devices-hap/meterplus.js.map +0 -1
  368. package/dist/devices-hap/meterpro.d.ts +0 -43
  369. package/dist/devices-hap/meterpro.d.ts.map +0 -1
  370. package/dist/devices-hap/meterpro.js +0 -468
  371. package/dist/devices-hap/meterpro.js.map +0 -1
  372. package/dist/devices-hap/motion.d.ts +0 -42
  373. package/dist/devices-hap/motion.d.ts.map +0 -1
  374. package/dist/devices-hap/motion.js +0 -345
  375. package/dist/devices-hap/motion.js.map +0 -1
  376. package/dist/devices-hap/plug.d.ts +0 -49
  377. package/dist/devices-hap/plug.d.ts.map +0 -1
  378. package/dist/devices-hap/plug.js +0 -395
  379. package/dist/devices-hap/plug.js.map +0 -1
  380. package/dist/devices-hap/relayswitch.d.ts +0 -96
  381. package/dist/devices-hap/relayswitch.d.ts.map +0 -1
  382. package/dist/devices-hap/relayswitch.js +0 -642
  383. package/dist/devices-hap/relayswitch.js.map +0 -1
  384. package/dist/devices-hap/robotvacuumcleaner.d.ts +0 -54
  385. package/dist/devices-hap/robotvacuumcleaner.d.ts.map +0 -1
  386. package/dist/devices-hap/robotvacuumcleaner.js +0 -523
  387. package/dist/devices-hap/robotvacuumcleaner.js.map +0 -1
  388. package/dist/devices-hap/waterdetector.d.ts +0 -41
  389. package/dist/devices-hap/waterdetector.d.ts.map +0 -1
  390. package/dist/devices-hap/waterdetector.js +0 -356
  391. package/dist/devices-hap/waterdetector.js.map +0 -1
  392. package/dist/devices-matter/BaseMatterAccessory.d.ts +0 -78
  393. package/dist/devices-matter/BaseMatterAccessory.d.ts.map +0 -1
  394. package/dist/devices-matter/BaseMatterAccessory.js +0 -244
  395. package/dist/devices-matter/BaseMatterAccessory.js.map +0 -1
  396. package/dist/devices-matter/ColorLightAccessory.d.ts +0 -20
  397. package/dist/devices-matter/ColorLightAccessory.d.ts.map +0 -1
  398. package/dist/devices-matter/ColorLightAccessory.js +0 -95
  399. package/dist/devices-matter/ColorLightAccessory.js.map +0 -1
  400. package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts +0 -18
  401. package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts.map +0 -1
  402. package/dist/devices-matter/ColorTemperatureLightAccessory.js +0 -76
  403. package/dist/devices-matter/ColorTemperatureLightAccessory.js.map +0 -1
  404. package/dist/devices-matter/ContactSensorAccessory.d.ts +0 -12
  405. package/dist/devices-matter/ContactSensorAccessory.d.ts.map +0 -1
  406. package/dist/devices-matter/ContactSensorAccessory.js +0 -34
  407. package/dist/devices-matter/ContactSensorAccessory.js.map +0 -1
  408. package/dist/devices-matter/DimmableLightAccessory.d.ts +0 -58
  409. package/dist/devices-matter/DimmableLightAccessory.d.ts.map +0 -1
  410. package/dist/devices-matter/DimmableLightAccessory.js +0 -167
  411. package/dist/devices-matter/DimmableLightAccessory.js.map +0 -1
  412. package/dist/devices-matter/DoorLockAccessory.d.ts +0 -14
  413. package/dist/devices-matter/DoorLockAccessory.d.ts.map +0 -1
  414. package/dist/devices-matter/DoorLockAccessory.js +0 -50
  415. package/dist/devices-matter/DoorLockAccessory.js.map +0 -1
  416. package/dist/devices-matter/ExtendedColorLightAccessory.d.ts +0 -21
  417. package/dist/devices-matter/ExtendedColorLightAccessory.d.ts.map +0 -1
  418. package/dist/devices-matter/ExtendedColorLightAccessory.js +0 -106
  419. package/dist/devices-matter/ExtendedColorLightAccessory.js.map +0 -1
  420. package/dist/devices-matter/FanAccessory.d.ts +0 -16
  421. package/dist/devices-matter/FanAccessory.d.ts.map +0 -1
  422. package/dist/devices-matter/FanAccessory.js +0 -81
  423. package/dist/devices-matter/FanAccessory.js.map +0 -1
  424. package/dist/devices-matter/HumiditySensorAccessory.d.ts +0 -12
  425. package/dist/devices-matter/HumiditySensorAccessory.d.ts.map +0 -1
  426. package/dist/devices-matter/HumiditySensorAccessory.js +0 -34
  427. package/dist/devices-matter/HumiditySensorAccessory.js.map +0 -1
  428. package/dist/devices-matter/LeakSensorAccessory.d.ts +0 -12
  429. package/dist/devices-matter/LeakSensorAccessory.d.ts.map +0 -1
  430. package/dist/devices-matter/LeakSensorAccessory.js +0 -33
  431. package/dist/devices-matter/LeakSensorAccessory.js.map +0 -1
  432. package/dist/devices-matter/LightSensorAccessory.d.ts +0 -12
  433. package/dist/devices-matter/LightSensorAccessory.d.ts.map +0 -1
  434. package/dist/devices-matter/LightSensorAccessory.js +0 -34
  435. package/dist/devices-matter/LightSensorAccessory.js.map +0 -1
  436. package/dist/devices-matter/OccupancySensorAccessory.d.ts +0 -12
  437. package/dist/devices-matter/OccupancySensorAccessory.d.ts.map +0 -1
  438. package/dist/devices-matter/OccupancySensorAccessory.js +0 -39
  439. package/dist/devices-matter/OccupancySensorAccessory.js.map +0 -1
  440. package/dist/devices-matter/OnOffLightAccessory.d.ts +0 -38
  441. package/dist/devices-matter/OnOffLightAccessory.d.ts.map +0 -1
  442. package/dist/devices-matter/OnOffLightAccessory.js +0 -110
  443. package/dist/devices-matter/OnOffLightAccessory.js.map +0 -1
  444. package/dist/devices-matter/OnOffOutletAccessory.d.ts +0 -14
  445. package/dist/devices-matter/OnOffOutletAccessory.d.ts.map +0 -1
  446. package/dist/devices-matter/OnOffOutletAccessory.js +0 -43
  447. package/dist/devices-matter/OnOffOutletAccessory.js.map +0 -1
  448. package/dist/devices-matter/OnOffSwitchAccessory.d.ts +0 -14
  449. package/dist/devices-matter/OnOffSwitchAccessory.d.ts.map +0 -1
  450. package/dist/devices-matter/OnOffSwitchAccessory.js +0 -42
  451. package/dist/devices-matter/OnOffSwitchAccessory.js.map +0 -1
  452. package/dist/devices-matter/RoboticVacuumAccessory.d.ts +0 -68
  453. package/dist/devices-matter/RoboticVacuumAccessory.d.ts.map +0 -1
  454. package/dist/devices-matter/RoboticVacuumAccessory.js +0 -334
  455. package/dist/devices-matter/RoboticVacuumAccessory.js.map +0 -1
  456. package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts +0 -11
  457. package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts.map +0 -1
  458. package/dist/devices-matter/SmokeCOAlarmAccessory.js +0 -49
  459. package/dist/devices-matter/SmokeCOAlarmAccessory.js.map +0 -1
  460. package/dist/devices-matter/TemperatureSensorAccessory.d.ts +0 -12
  461. package/dist/devices-matter/TemperatureSensorAccessory.d.ts.map +0 -1
  462. package/dist/devices-matter/TemperatureSensorAccessory.js +0 -36
  463. package/dist/devices-matter/TemperatureSensorAccessory.js.map +0 -1
  464. package/dist/devices-matter/ThermostatAccessory.d.ts +0 -19
  465. package/dist/devices-matter/ThermostatAccessory.d.ts.map +0 -1
  466. package/dist/devices-matter/ThermostatAccessory.js +0 -95
  467. package/dist/devices-matter/ThermostatAccessory.js.map +0 -1
  468. package/dist/devices-matter/VenetianBlindAccessory.d.ts +0 -19
  469. package/dist/devices-matter/VenetianBlindAccessory.d.ts.map +0 -1
  470. package/dist/devices-matter/VenetianBlindAccessory.js +0 -99
  471. package/dist/devices-matter/VenetianBlindAccessory.js.map +0 -1
  472. package/dist/devices-matter/WindowBlindAccessory.d.ts +0 -17
  473. package/dist/devices-matter/WindowBlindAccessory.d.ts.map +0 -1
  474. package/dist/devices-matter/WindowBlindAccessory.js +0 -80
  475. package/dist/devices-matter/WindowBlindAccessory.js.map +0 -1
  476. package/dist/devices-matter/custom/PowerStripAccessory.d.ts +0 -97
  477. package/dist/devices-matter/custom/PowerStripAccessory.d.ts.map +0 -1
  478. package/dist/devices-matter/custom/PowerStripAccessory.js +0 -265
  479. package/dist/devices-matter/custom/PowerStripAccessory.js.map +0 -1
  480. package/dist/devices-matter/custom/index.d.ts +0 -8
  481. package/dist/devices-matter/custom/index.d.ts.map +0 -1
  482. package/dist/devices-matter/custom/index.js +0 -8
  483. package/dist/devices-matter/custom/index.js.map +0 -1
  484. package/dist/devices-matter/index.d.ts +0 -29
  485. package/dist/devices-matter/index.d.ts.map +0 -1
  486. package/dist/devices-matter/index.js +0 -28
  487. package/dist/devices-matter/index.js.map +0 -1
  488. package/dist/index.test.d.ts +0 -2
  489. package/dist/index.test.d.ts.map +0 -1
  490. package/dist/index.test.js +0 -19
  491. package/dist/index.test.js.map +0 -1
  492. package/dist/irdevice/airconditioner.d.ts +0 -61
  493. package/dist/irdevice/airconditioner.d.ts.map +0 -1
  494. package/dist/irdevice/airconditioner.js +0 -472
  495. package/dist/irdevice/airconditioner.js.map +0 -1
  496. package/dist/irdevice/airpurifier.d.ts +0 -50
  497. package/dist/irdevice/airpurifier.d.ts.map +0 -1
  498. package/dist/irdevice/airpurifier.js +0 -213
  499. package/dist/irdevice/airpurifier.js.map +0 -1
  500. package/dist/irdevice/camera.d.ts +0 -32
  501. package/dist/irdevice/camera.d.ts.map +0 -1
  502. package/dist/irdevice/camera.js +0 -107
  503. package/dist/irdevice/camera.js.map +0 -1
  504. package/dist/irdevice/fan.d.ts +0 -36
  505. package/dist/irdevice/fan.d.ts.map +0 -1
  506. package/dist/irdevice/fan.js +0 -200
  507. package/dist/irdevice/fan.js.map +0 -1
  508. package/dist/irdevice/irdevice.d.ts +0 -68
  509. package/dist/irdevice/irdevice.d.ts.map +0 -1
  510. package/dist/irdevice/irdevice.js +0 -398
  511. package/dist/irdevice/irdevice.js.map +0 -1
  512. package/dist/irdevice/light.d.ts +0 -36
  513. package/dist/irdevice/light.d.ts.map +0 -1
  514. package/dist/irdevice/light.js +0 -206
  515. package/dist/irdevice/light.js.map +0 -1
  516. package/dist/irdevice/other.d.ts +0 -57
  517. package/dist/irdevice/other.d.ts.map +0 -1
  518. package/dist/irdevice/other.js +0 -778
  519. package/dist/irdevice/other.js.map +0 -1
  520. package/dist/irdevice/tv.d.ts +0 -45
  521. package/dist/irdevice/tv.d.ts.map +0 -1
  522. package/dist/irdevice/tv.js +0 -327
  523. package/dist/irdevice/tv.js.map +0 -1
  524. package/dist/irdevice/vacuumcleaner.d.ts +0 -28
  525. package/dist/irdevice/vacuumcleaner.d.ts.map +0 -1
  526. package/dist/irdevice/vacuumcleaner.js +0 -104
  527. package/dist/irdevice/vacuumcleaner.js.map +0 -1
  528. package/dist/irdevice/waterheater.d.ts +0 -30
  529. package/dist/irdevice/waterheater.d.ts.map +0 -1
  530. package/dist/irdevice/waterheater.js +0 -105
  531. package/dist/irdevice/waterheater.js.map +0 -1
  532. package/dist/platform-hap.d.ts +0 -145
  533. package/dist/platform-hap.d.ts.map +0 -1
  534. package/dist/platform-hap.js +0 -2823
  535. package/dist/platform-hap.js.map +0 -1
  536. package/dist/platform-matter.d.ts +0 -131
  537. package/dist/platform-matter.d.ts.map +0 -1
  538. package/dist/platform-matter.js +0 -1002
  539. package/dist/platform-matter.js.map +0 -1
  540. package/dist/utils.test.d.ts +0 -2
  541. package/dist/utils.test.d.ts.map +0 -1
  542. package/dist/utils.test.js +0 -95
  543. package/dist/utils.test.js.map +0 -1
  544. package/dist/verifyconfig.test.d.ts +0 -2
  545. package/dist/verifyconfig.test.d.ts.map +0 -1
  546. package/dist/verifyconfig.test.js +0 -167
  547. package/dist/verifyconfig.test.js.map +0 -1
  548. package/src/baseMatterAccessory.test.ts +0 -88
  549. package/src/custom.d.ts +0 -7
  550. package/src/devices-hap/airpurifier.ts +0 -563
  551. package/src/devices-hap/blindtilt.ts +0 -1049
  552. package/src/devices-hap/bot.ts +0 -900
  553. package/src/devices-hap/ceilinglight.ts +0 -742
  554. package/src/devices-hap/colorbulb.ts +0 -904
  555. package/src/devices-hap/contact.ts +0 -457
  556. package/src/devices-hap/curtain.ts +0 -944
  557. package/src/devices-hap/device.ts +0 -884
  558. package/src/devices-hap/fan.ts +0 -711
  559. package/src/devices-hap/hub.ts +0 -439
  560. package/src/devices-hap/humidifier.ts +0 -669
  561. package/src/devices-hap/iosensor.ts +0 -427
  562. package/src/devices-hap/lightstrip.ts +0 -836
  563. package/src/devices-hap/lock.ts +0 -620
  564. package/src/devices-hap/meter.ts +0 -426
  565. package/src/devices-hap/meterplus.ts +0 -430
  566. package/src/devices-hap/meterpro.ts +0 -522
  567. package/src/devices-hap/motion.ts +0 -390
  568. package/src/devices-hap/plug.ts +0 -423
  569. package/src/devices-hap/relayswitch.ts +0 -727
  570. package/src/devices-hap/robotvacuumcleaner.ts +0 -568
  571. package/src/devices-hap/waterdetector.ts +0 -400
  572. package/src/devices-matter/BaseMatterAccessory.ts +0 -273
  573. package/src/devices-matter/ColorLightAccessory.ts +0 -110
  574. package/src/devices-matter/ColorTemperatureLightAccessory.ts +0 -90
  575. package/src/devices-matter/ContactSensorAccessory.ts +0 -41
  576. package/src/devices-matter/DimmableLightAccessory.ts +0 -192
  577. package/src/devices-matter/DoorLockAccessory.ts +0 -60
  578. package/src/devices-matter/ExtendedColorLightAccessory.ts +0 -122
  579. package/src/devices-matter/FanAccessory.ts +0 -95
  580. package/src/devices-matter/HumiditySensorAccessory.ts +0 -41
  581. package/src/devices-matter/LeakSensorAccessory.ts +0 -40
  582. package/src/devices-matter/LightSensorAccessory.ts +0 -41
  583. package/src/devices-matter/OccupancySensorAccessory.ts +0 -48
  584. package/src/devices-matter/OnOffLightAccessory.ts +0 -125
  585. package/src/devices-matter/OnOffOutletAccessory.ts +0 -51
  586. package/src/devices-matter/OnOffSwitchAccessory.ts +0 -51
  587. package/src/devices-matter/RoboticVacuumAccessory.ts +0 -407
  588. package/src/devices-matter/SmokeCOAlarmAccessory.ts +0 -59
  589. package/src/devices-matter/TemperatureSensorAccessory.ts +0 -43
  590. package/src/devices-matter/ThermostatAccessory.ts +0 -110
  591. package/src/devices-matter/VenetianBlindAccessory.ts +0 -115
  592. package/src/devices-matter/WindowBlindAccessory.ts +0 -92
  593. package/src/devices-matter/custom/PowerStripAccessory.ts +0 -309
  594. package/src/devices-matter/custom/index.ts +0 -8
  595. package/src/devices-matter/index.ts +0 -29
  596. package/src/index.test.ts +0 -24
  597. package/src/irdevice/airconditioner.ts +0 -533
  598. package/src/irdevice/airpurifier.ts +0 -252
  599. package/src/irdevice/camera.ts +0 -129
  600. package/src/irdevice/fan.ts +0 -226
  601. package/src/irdevice/irdevice.ts +0 -435
  602. package/src/irdevice/light.ts +0 -246
  603. package/src/irdevice/other.ts +0 -790
  604. package/src/irdevice/tv.ts +0 -378
  605. package/src/irdevice/vacuumcleaner.ts +0 -126
  606. package/src/irdevice/waterheater.ts +0 -127
  607. package/src/platform-hap.ts +0 -2952
  608. package/src/platform-matter.ts +0 -1129
  609. package/src/utils.test.ts +0 -96
  610. package/src/verifyconfig.test.ts +0 -198
@@ -1,944 +0,0 @@
1
- /* Copyright(C) 2021-2024, donavanbecker (https://github.com/donavanbecker). All rights reserved.
2
- *
3
- * curtain.ts: @switchbot/homebridge-switchbot.
4
- */
5
- import type { CharacteristicChange, CharacteristicValue, PlatformAccessory, Service } from 'homebridge'
6
- import type { bodyChange, curtain3ServiceData, curtain3WebhookContext, curtainServiceData, curtainStatus, curtainWebhookContext, device, SwitchBotBLE, SwitchbotDevice, WoCurtain } from 'node-switchbot'
7
-
8
- import type { SwitchBotHAPPlatform } from '../platform-hap.js'
9
- import type { curtainConfig, devicesConfig } from '../settings.js'
10
-
11
- import { hostname } from 'node:os'
12
-
13
- /*
14
- * For Testing Locally:
15
- * import { SwitchBotBLEModel, SwitchBotBLEModelName } from '/Users/Shared/GitHub/OpenWonderLabs/node-switchbot/dist/index.js';
16
- */
17
- import { SwitchBotBLEModel, SwitchBotBLEModelName } from 'node-switchbot'
18
- import { debounceTime, interval, skipWhile, Subject, take, tap } from 'rxjs'
19
-
20
- import { formatDeviceIdAsMac, isCurtainDevice } from '../utils.js'
21
- import { deviceBase } from './device.js'
22
-
23
- export class Curtain extends deviceBase {
24
- // Services
25
- private WindowCovering: {
26
- Name: CharacteristicValue
27
- Service: Service
28
- PositionState: CharacteristicValue
29
- TargetPosition: CharacteristicValue
30
- CurrentPosition: CharacteristicValue
31
- HoldPosition: CharacteristicValue
32
- }
33
-
34
- private Battery: {
35
- Name: CharacteristicValue
36
- Service: Service
37
- BatteryLevel: CharacteristicValue
38
- StatusLowBattery: CharacteristicValue
39
- ChargingState: CharacteristicValue
40
- }
41
-
42
- private LightSensor?: {
43
- Name: CharacteristicValue
44
- Service: Service
45
- CurrentAmbientLightLevel?: CharacteristicValue
46
- }
47
-
48
- private OpenModeSwitch?: {
49
- Name: string
50
- Service: Service
51
- On: CharacteristicValue
52
- }
53
-
54
- private CloseModeSwitch?: {
55
- Name: string
56
- Service: Service
57
- On: CharacteristicValue
58
- }
59
-
60
- // OpenAPI
61
- deviceStatus!: curtainStatus
62
-
63
- // Webhook
64
- webhookContext!: curtainWebhookContext | curtain3WebhookContext
65
-
66
- // BLE
67
- serviceData!: curtainServiceData | curtain3ServiceData
68
-
69
- // Target
70
- hasLoggedStandby!: boolean
71
- setNewTarget!: boolean
72
- setNewTargetTimer!: NodeJS.Timeout
73
-
74
- // Updates
75
- curtainMoving!: boolean
76
- curtainUpdateInProgress!: boolean
77
- doCurtainUpdate!: Subject<void>
78
-
79
- constructor(
80
- readonly platform: SwitchBotHAPPlatform,
81
- accessory: PlatformAccessory,
82
- device: device & devicesConfig,
83
- ) {
84
- super(platform, accessory, device)
85
- // Set category
86
- accessory.category = this.hap.Categories.WINDOW_COVERING
87
-
88
- // this is subject we use to track when we need to POST changes to the SwitchBot API
89
- this.doCurtainUpdate = new Subject()
90
- this.curtainMoving = false
91
- this.curtainUpdateInProgress = false
92
- this.setNewTarget = false
93
-
94
- // Initialize WindowCovering Service
95
- accessory.context.WindowCovering = accessory.context.WindowCovering ?? {}
96
- this.WindowCovering = {
97
- Name: accessory.displayName,
98
- Service: accessory.getService(this.hap.Service.WindowCovering) ?? accessory.addService(this.hap.Service.WindowCovering) as Service,
99
- PositionState: accessory.context.PositionState ?? this.hap.Characteristic.PositionState.STOPPED,
100
- TargetPosition: accessory.context.TargetPosition ?? 100,
101
- CurrentPosition: accessory.context.CurrentPosition ?? 100,
102
- HoldPosition: accessory.context.HoldPosition ?? false,
103
- }
104
- accessory.context.WindowCovering = this.WindowCovering as object
105
-
106
- // Initialize WindowCovering Service
107
- this.WindowCovering.Service.setCharacteristic(this.hap.Characteristic.Name, this.WindowCovering.Name).setCharacteristic(this.hap.Characteristic.ObstructionDetected, false).getCharacteristic(this.hap.Characteristic.PositionState).onGet(() => {
108
- return this.WindowCovering.PositionState
109
- })
110
-
111
- // Initialize WindowCovering CurrentPosition
112
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.CurrentPosition).setProps({
113
- minStep: (device as curtainConfig).set_minStep ?? 1,
114
- minValue: 0,
115
- maxValue: 100,
116
- validValueRanges: [0, 100],
117
- }).onGet(() => {
118
- return this.WindowCovering.CurrentPosition
119
- })
120
-
121
- // Initialize WindowCovering TargetPosition
122
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.TargetPosition).setProps({
123
- minStep: (device as curtainConfig).set_minStep ?? 1,
124
- minValue: 0,
125
- maxValue: 100,
126
- validValueRanges: [0, 100],
127
- }).onGet(() => {
128
- return this.WindowCovering.TargetPosition
129
- }).onSet(this.TargetPositionSet.bind(this))
130
-
131
- // Initialize WindowCovering TargetPosition
132
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.HoldPosition).onGet(() => {
133
- return this.WindowCovering.HoldPosition
134
- }).onSet(this.HoldPositionSet.bind(this))
135
-
136
- // Initialize Battery Service
137
- accessory.context.Battery = accessory.context.Battery ?? {}
138
- this.Battery = {
139
- Name: `${accessory.displayName} Battery`,
140
- Service: accessory.getService(this.hap.Service.Battery) ?? accessory.addService(this.hap.Service.Battery) as Service,
141
- BatteryLevel: accessory.context.BatteryLevel ?? 100,
142
- StatusLowBattery: accessory.context.StatusLowBattery ?? this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_NORMAL,
143
- ChargingState: accessory.context.ChargingState ?? this.hap.Characteristic.ChargingState.NOT_CHARGING,
144
- }
145
- accessory.context.Battery = this.Battery as object
146
-
147
- // Initialize Battery Service
148
- this.Battery.Service.setCharacteristic(this.hap.Characteristic.Name, this.Battery.Name).getCharacteristic(this.hap.Characteristic.BatteryLevel).onGet(() => {
149
- return this.Battery.BatteryLevel
150
- })
151
-
152
- this.Battery.Service.getCharacteristic(this.hap.Characteristic.StatusLowBattery).onGet(() => {
153
- return this.Battery.StatusLowBattery
154
- })
155
-
156
- this.Battery.Service.getCharacteristic(this.hap.Characteristic.ChargingState).onGet(() => {
157
- return this.Battery.ChargingState
158
- })
159
-
160
- // Initialize LightSensor Service
161
- if ((device as curtainConfig).hide_lightsensor || !isCurtainDevice(device)) {
162
- if (this.LightSensor?.Service) {
163
- this.debugLog('Removing Light Sensor Service')
164
- this.LightSensor.Service = this.accessory.getService(this.hap.Service.LightSensor) as Service
165
- accessory.removeService(this.LightSensor.Service)
166
- accessory.context.LightSensor = {}
167
- }
168
- } else {
169
- accessory.context.LightSensor = accessory.context.LightSensor ?? {}
170
- this.LightSensor = {
171
- Name: `${accessory.displayName} Light Sensor`,
172
- Service: accessory.getService(this.hap.Service.LightSensor) ?? this.accessory.addService(this.hap.Service.LightSensor) as Service,
173
- CurrentAmbientLightLevel: accessory.context.CurrentAmbientLightLevel ?? 0.0001,
174
- }
175
- accessory.context.LightSensor = this.LightSensor as object
176
-
177
- // Initialize LightSensor Characteristic
178
- this.LightSensor.Service.setCharacteristic(this.hap.Characteristic.Name, this.LightSensor.Name).setCharacteristic(this.hap.Characteristic.StatusActive, true).getCharacteristic(this.hap.Characteristic.CurrentAmbientLightLevel).onGet(() => {
179
- return this.LightSensor!.CurrentAmbientLightLevel!
180
- })
181
- }
182
-
183
- // Initialize Open Mode Switch Service
184
- if (!(device as curtainConfig).silentModeSwitch) {
185
- if (this.OpenModeSwitch?.Service) {
186
- this.debugLog('Removing Open Mode Switch Service')
187
- this.OpenModeSwitch.Service = this.accessory.getService(this.hap.Service.Switch) as Service
188
- accessory.removeService(this.OpenModeSwitch.Service)
189
- accessory.context.OpenModeSwitch = {}
190
- }
191
- } else {
192
- accessory.context.OpenModeSwitch = accessory.context.OpenModeSwitch ?? {}
193
- this.debugLog('Adding Open Mode Switch Service')
194
- const name = `${accessory.displayName} Silent Open Mode`
195
- const uuid = this.api.hap.uuid.generate(name)
196
- this.OpenModeSwitch = {
197
- Name: name,
198
- Service: accessory.getService(name) ?? accessory.addService(this.hap.Service.Switch, name, uuid) as Service,
199
- On: accessory.context.OpenModeSwitch.On ?? false,
200
- }
201
- accessory.context.OpenModeSwitch = this.OpenModeSwitch as object
202
-
203
- // Initialize Open Mode Switch Service
204
- this.OpenModeSwitch.Service.setCharacteristic(this.hap.Characteristic.Name, this.OpenModeSwitch.Name).setCharacteristic(this.hap.Characteristic.ConfiguredName, this.OpenModeSwitch.Name).getCharacteristic(this.hap.Characteristic.On).onGet(() => {
205
- return this.OpenModeSwitch?.On ?? false
206
- })
207
-
208
- this.OpenModeSwitch.Service.getCharacteristic(this.hap.Characteristic.On).onSet(this.OpenModeSwitchSet.bind(this))
209
- }
210
-
211
- // Initialize Close Mode Switch Service
212
- if (!(device as curtainConfig).silentModeSwitch) {
213
- if (this.CloseModeSwitch?.Service) {
214
- this.debugLog('Removing Close Mode Switch Service')
215
- this.CloseModeSwitch.Service = this.accessory.getService(this.hap.Service.Switch) as Service
216
- accessory.removeService(this.CloseModeSwitch.Service)
217
- accessory.context.CloseModeSwitch = {}
218
- }
219
- } else {
220
- accessory.context.CloseModeSwitch = accessory.context.CloseModeSwitch ?? {}
221
- this.debugLog('Adding Close Mode Switch Service')
222
- const name = `${accessory.displayName} Silent Close Mode`
223
- const uuid = this.api.hap.uuid.generate(name)
224
- this.CloseModeSwitch = {
225
- Name: name,
226
- Service: this.accessory.getService(name) ?? accessory.addService(this.hap.Service.Switch, name, uuid) as Service,
227
- On: accessory.context.CloseModeSwitch.On ?? false,
228
- }
229
- accessory.context.CloseModeSwitch = this.CloseModeSwitch as object
230
-
231
- // Initialize Close Mode Switch Service
232
- this.CloseModeSwitch.Service.setCharacteristic(this.hap.Characteristic.Name, this.CloseModeSwitch.Name).setCharacteristic(this.hap.Characteristic.ConfiguredName, this.CloseModeSwitch.Name).getCharacteristic(this.hap.Characteristic.On).onGet(() => {
233
- return this.CloseModeSwitch?.On ?? false
234
- })
235
-
236
- this.CloseModeSwitch.Service.getCharacteristic(this.hap.Characteristic.On).onSet(this.CloseModeSwitchSet.bind(this))
237
- }
238
-
239
- // Retrieve initial values and updateHomekit
240
- try {
241
- this.debugLog('Retrieve initial values and update Homekit')
242
- this.refreshStatus()
243
- } catch (e: any) {
244
- this.errorLog(`failed to retrieve initial values and update Homekit, Error: ${e.message ?? e}`)
245
- }
246
-
247
- // regisiter webhook event handler if enabled
248
- try {
249
- this.debugLog('Registering Webhook Event Handler')
250
- this.registerWebhook()
251
- } catch (e: any) {
252
- this.errorLog(`failed to registerWebhook, Error: ${e.message ?? e}`)
253
- }
254
-
255
- // regisiter platform BLE event handler if enabled
256
- try {
257
- this.debugLog('Registering Platform BLE Event Handler')
258
- this.registerPlatformBLE()
259
- } catch (e: any) {
260
- this.errorLog(`failed to registerPlatformBLE, Error: ${e.message ?? e}`)
261
- }
262
-
263
- // History
264
- this.history()
265
-
266
- // Start an update interval
267
- interval(this.deviceRefreshRate * 1000)
268
- .pipe(skipWhile(() => this.curtainUpdateInProgress))
269
- .subscribe(async () => {
270
- await this.refreshStatus()
271
- })
272
-
273
- // update slide progress
274
- interval(this.deviceUpdateRate * 1000)
275
- .pipe(skipWhile(() => !this.curtainMoving))
276
- .subscribe(async () => {
277
- if (this.WindowCovering.PositionState === this.hap.Characteristic.PositionState.STOPPED) {
278
- return
279
- }
280
- this.debugLog(`Refresh Status When Moving, PositionState: ${this.WindowCovering.PositionState}`)
281
- await this.refreshStatus()
282
- })
283
-
284
- // Watch for Curtain change events
285
- // We put in a debounce of 100ms so we don't make duplicate calls
286
- this.doCurtainUpdate
287
- .pipe(
288
- tap(() => {
289
- this.curtainUpdateInProgress = true
290
- }),
291
- debounceTime(this.devicePushRate * 1000),
292
- )
293
- .subscribe(async () => {
294
- try {
295
- await this.pushChanges()
296
- } catch (e: any) {
297
- await this.apiError(e)
298
- this.errorLog(`failed pushChanges with ${device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`)
299
- }
300
- this.curtainUpdateInProgress = false
301
- })
302
-
303
- // Setup EVE history features
304
- this.setupHistoryService()
305
- }
306
-
307
- async history() {
308
- if (this.device.history === true) {
309
- // initialize when this accessory is newly created.
310
- this.accessory.context.lastActivation = this.accessory.context.lastActivation ?? 0
311
- } else {
312
- // removes cached values if history is turned off
313
- delete this.accessory.context.lastActivation
314
- }
315
- }
316
-
317
- /*
318
- * Setup EVE history features for curtain devices.
319
- */
320
- async setupHistoryService(): Promise<void> {
321
- if (this.device.history !== true) {
322
- return
323
- }
324
-
325
- try {
326
- const formattedDeviceId = formatDeviceIdAsMac(this.device.deviceId)
327
- this.device.bleMac = formattedDeviceId
328
- this.debugLog(`bleMac: ${this.device.bleMac}`)
329
- this.historyService = new this.platform.fakegatoAPI('custom', this.accessory, {
330
- log: this.platform.log,
331
- storage: 'fs',
332
- filename: `${hostname().split('.')[0]}_${this.device.bleMac}_persist.json`,
333
- })
334
- const motion: Service
335
- = this.accessory.getService(this.hap.Service.MotionSensor)
336
- || this.accessory.addService(this.hap.Service.MotionSensor, 'Motion')
337
- motion.addOptionalCharacteristic(this.platform.eve.Characteristics.LastActivation)
338
- motion.getCharacteristic(this.platform.eve.Characteristics.LastActivation).onGet(() => {
339
- const lastActivation = this.accessory.context.lastActivation
340
- ? Math.max(0, this.accessory.context.lastActivation - this.historyService.getInitialTime())
341
- : 0
342
- return lastActivation
343
- })
344
- await this.setMinMax()
345
- motion.getCharacteristic(this.hap.Characteristic.MotionDetected).on('change', (event: CharacteristicChange) => {
346
- if (event.newValue !== event.oldValue) {
347
- const sensor = this.accessory.getService(this.hap.Service.MotionSensor)
348
- const entry = {
349
- time: Math.round(new Date().valueOf() / 1000),
350
- motion: event.newValue,
351
- }
352
- this.accessory.context.lastActivation = entry.time
353
- sensor?.updateCharacteristic(
354
- this.platform.eve.Characteristics.LastActivation,
355
- Math.max(0, this.accessory.context.lastActivation - this.historyService.getInitialTime()),
356
- )
357
- this.historyService.addEntry(entry)
358
- }
359
- })
360
- this.updateHistory()
361
- } catch (error) {
362
- this.errorLog(`failed to format device ID as MAC, Error: ${error}`)
363
- }
364
- }
365
-
366
- updateHistory(): void {
367
- const motion = Number(this.WindowCovering.CurrentPosition) > 0 ? 1 : 0
368
- this.historyService.addEntry({
369
- time: Math.round(new Date().valueOf() / 1000),
370
- motion,
371
- })
372
- setTimeout(() => {
373
- this.updateHistory()
374
- }, 10 * 60 * 1000)
375
- }
376
-
377
- async BLEparseStatus(): Promise<void> {
378
- this.debugLog('BLEparseStatus')
379
- this.debugLog(`(position, battery) = BLE:(${this.serviceData.position}, ${this.serviceData.battery}), current:(${this.WindowCovering.CurrentPosition}, ${this.Battery.BatteryLevel})`)
380
- // CurrentPosition
381
- if ('position' in this.serviceData && this.serviceData.position !== undefined && this.serviceData.position !== null && !Number.isNaN(Number(this.serviceData.position))) {
382
- this.WindowCovering.CurrentPosition = 100 - Number(this.serviceData.position)
383
- await this.getCurrentPostion()
384
- } else {
385
- this.warnLog(`Invalid position data from BLE: ${this.serviceData.position}`)
386
- }
387
- // CurrentAmbientLightLevel
388
- if (!(this.device as curtainConfig).hide_lightsensor && this.LightSensor?.Service && 'lightLevel' in this.serviceData) {
389
- const set_minLux = (this.device as curtainConfig).set_minLux ?? 1
390
- const set_maxLux = (this.device as curtainConfig).set_maxLux ?? 6001
391
- const lightLevel = this.serviceData.lightLevel
392
- this.LightSensor.CurrentAmbientLightLevel = this.getLightLevel(lightLevel, set_minLux, set_maxLux, 19)
393
- this.debugLog(`LightLevel: ${this.serviceData.lightLevel}, CurrentAmbientLightLevel: ${this.LightSensor.CurrentAmbientLightLevel}`)
394
- }
395
- // Battery Info
396
- if ('battery' in this.serviceData) {
397
- // BatteryLevel
398
- this.Battery.BatteryLevel = this.serviceData.battery
399
- this.debugLog(`BatteryLevel: ${this.Battery.BatteryLevel}`)
400
- // StatusLowBattery
401
- this.Battery.StatusLowBattery = this.Battery.BatteryLevel < 10
402
- ? this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_LOW
403
- : this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_NORMAL
404
- this.debugLog(`StatusLowBattery: ${this.Battery.StatusLowBattery}`)
405
- }
406
- }
407
-
408
- async openAPIparseStatus(): Promise<void> {
409
- this.debugLog('openAPIparseStatus')
410
- this.debugLog(`(slidePosition, battery, version) = OpenAPI:(${this.deviceStatus.slidePosition}, ${this.deviceStatus.battery}, ${this.deviceStatus.version}), current:(${this.WindowCovering.CurrentPosition}, ${this.Battery.BatteryLevel}, ${this.accessory.context.version})`)
411
- // CurrentPosition
412
- this.WindowCovering.CurrentPosition = 100 - this.deviceStatus.slidePosition
413
- await this.getCurrentPostion()
414
-
415
- // Brightness
416
- if (!(this.device as curtainConfig).hide_lightsensor && this.LightSensor?.Service) {
417
- const set_minLux = (this.device as curtainConfig).set_minLux ?? 1
418
- const set_maxLux = (this.device as curtainConfig).set_maxLux ?? 6001
419
- const lightLevel = this.deviceStatus.lightLevel === 'bright' ? set_maxLux : set_minLux
420
- this.LightSensor.CurrentAmbientLightLevel = this.getLightLevel(lightLevel, set_minLux, set_maxLux, 2)
421
- this.debugLog(`CurrentAmbientLightLevel: ${this.LightSensor.CurrentAmbientLightLevel}`)
422
- }
423
-
424
- // BatteryLevel
425
- this.Battery.BatteryLevel = this.deviceStatus.battery
426
- this.debugLog(`BatteryLevel: ${this.Battery.BatteryLevel}`)
427
-
428
- // StatusLowBattery
429
- this.Battery.StatusLowBattery = this.Battery.BatteryLevel < 10
430
- ? this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_LOW
431
- : this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_NORMAL
432
- this.debugLog(`StatusLowBattery: ${this.Battery.StatusLowBattery}`)
433
-
434
- // Firmware Version
435
- if (this.deviceStatus.version) {
436
- const version = this.deviceStatus.version.toString()
437
- this.debugLog(`Firmware Version: ${version.replace(/^V|-.*$/g, '')}`)
438
- const deviceVersion = version.replace(/^V|-.*$/g, '') ?? '0.0.0'
439
- this.accessory
440
- .getService(this.hap.Service.AccessoryInformation)!
441
- .setCharacteristic(this.hap.Characteristic.HardwareRevision, deviceVersion)
442
- .setCharacteristic(this.hap.Characteristic.FirmwareRevision, deviceVersion)
443
- .getCharacteristic(this.hap.Characteristic.FirmwareRevision)
444
- .updateValue(deviceVersion)
445
- this.accessory.context.version = deviceVersion
446
- this.debugSuccessLog(`version: ${this.accessory.context.version}`)
447
- }
448
- }
449
-
450
- async parseStatusWebhook(): Promise<void> {
451
- this.debugLog('parseStatusWebhook')
452
- this.debugLog(`(slidePosition, battery) = Webhook:(${this.webhookContext.slidePosition}, ${this.webhookContext.battery}), current:(${this.WindowCovering.CurrentPosition}, ${this.Battery.BatteryLevel})`)
453
-
454
- // CurrentPosition
455
- this.WindowCovering.CurrentPosition = 100 - this.webhookContext.slidePosition
456
- await this.getCurrentPostion()
457
-
458
- // BatteryLevel
459
- this.Battery.BatteryLevel = this.webhookContext.battery
460
- this.debugLog(`BatteryLevel: ${this.Battery.BatteryLevel}`)
461
-
462
- // StatusLowBattery
463
- this.Battery.StatusLowBattery = this.Battery.BatteryLevel < 10
464
- ? this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_LOW
465
- : this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_NORMAL
466
- this.debugLog(`StatusLowBattery: ${this.Battery.StatusLowBattery}`)
467
- }
468
-
469
- /**
470
- * Asks the SwitchBot API for the latest device information
471
- */
472
- async refreshStatus(): Promise<void> {
473
- if (this.BLE) {
474
- await this.BLERefreshStatus()
475
- } else if (this.OpenAPI && this.platform.config.credentials?.token) {
476
- await this.openAPIRefreshStatus()
477
- } else {
478
- await this.offlineOff()
479
- this.debugWarnLog(`Connection Type: ${this.device.connectionType}, refreshStatus will not happen.`)
480
- }
481
- }
482
-
483
- async BLERefreshStatus(): Promise<void> {
484
- this.debugLog('BLERefreshStatus')
485
- const switchBotBLE = await this.switchbotBLE()
486
- if (switchBotBLE === undefined) {
487
- await this.BLERefreshConnection(switchBotBLE)
488
- } else {
489
- // Start to monitor advertisement packets
490
- (async () => {
491
- // Start to monitor advertisement packets
492
- const serviceData = await this.monitorAdvertisementPackets(switchBotBLE) as curtainServiceData | curtain3ServiceData
493
- // Update HomeKit
494
- if ((serviceData.model === SwitchBotBLEModel.Curtain || serviceData.model === SwitchBotBLEModel.Curtain3)
495
- && (serviceData.modelName === SwitchBotBLEModelName.Curtain || serviceData.modelName === SwitchBotBLEModelName.Curtain3)) {
496
- this.serviceData = serviceData
497
- if (serviceData !== undefined && serviceData !== null) {
498
- await this.BLEparseStatus()
499
- await this.updateHomeKitCharacteristics()
500
- } else {
501
- this.errorLog(`serviceData is either undefined or null, serviceData: ${JSON.stringify(serviceData)}`)
502
- await this.BLERefreshConnection(switchBotBLE)
503
- }
504
- } else {
505
- this.errorLog(`failed to get serviceData, serviceData: ${JSON.stringify(serviceData)}`)
506
- await this.BLERefreshConnection(switchBotBLE)
507
- }
508
- })()
509
- }
510
- }
511
-
512
- async openAPIRefreshStatus(): Promise<void> {
513
- this.debugLog('openAPIRefreshStatus')
514
- try {
515
- const deviceStatus = await this.deviceRefreshStatus<curtainStatus>()
516
- this.debugLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
517
- if (await this.successfulStatusCodes(deviceStatus)) {
518
- this.debugSuccessLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
519
- this.deviceStatus = deviceStatus.body
520
- await this.openAPIparseStatus()
521
- await this.updateHomeKitCharacteristics()
522
- } else {
523
- this.debugWarnLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
524
- }
525
- } catch (e: any) {
526
- await this.apiError(e)
527
- this.errorLog(`failed openAPIRefreshStatus with ${this.device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`)
528
- }
529
- }
530
-
531
- async registerWebhook() {
532
- if (this.device.webhook) {
533
- this.debugLog('is listening webhook.')
534
- this.platform.webhookEventHandler[this.device.deviceId] = async (context: curtainWebhookContext | curtain3WebhookContext) => {
535
- try {
536
- this.webhookContext = context
537
- if (context !== undefined || context !== null) {
538
- this.debugLog(`received Webhook: ${JSON.stringify(context)}`)
539
- await this.parseStatusWebhook()
540
- await this.updateHomeKitCharacteristics()
541
- } else {
542
- this.errorLog(`context is either undefined or null, context: ${JSON.stringify(context)}`)
543
- }
544
- } catch (e: any) {
545
- this.errorLog(`failed to handle webhook. Received: ${JSON.stringify(context)} Error: ${e.message ?? e}`)
546
- }
547
- }
548
- } else {
549
- this.debugLog('is not listening webhook.')
550
- }
551
- }
552
-
553
- async registerPlatformBLE(): Promise<void> {
554
- this.debugLog('registerPlatformBLE')
555
- if (this.config.options?.BLE && !this.device.disablePlatformBLE) {
556
- this.debugLog('is listening to Platform BLE.')
557
- try {
558
- const formattedDeviceId = formatDeviceIdAsMac(this.device.deviceId)
559
- this.device.bleMac = formattedDeviceId
560
- this.debugLog(`bleMac: ${this.device.bleMac}`)
561
- this.platform.bleEventHandler[this.device.bleMac] = async (context: curtainServiceData | curtain3ServiceData) => {
562
- try {
563
- this.serviceData = context
564
- if (context !== undefined && context !== null) {
565
- this.debugLog(`received BLE: ${JSON.stringify(context)}`)
566
- await this.BLEparseStatus()
567
- await this.updateHomeKitCharacteristics()
568
- } else {
569
- this.errorLog(`context is either undefined or null, context: ${JSON.stringify(context)}`)
570
- await this.BLERefreshConnection(context)
571
- }
572
- } catch (e: any) {
573
- this.errorLog(`failed to handle BLE. Received: ${JSON.stringify(context)} Error: ${e.message ?? e}`)
574
- }
575
- }
576
- } catch (error) {
577
- this.errorLog(`failed to format device ID as MAC, Error: ${error}`)
578
- }
579
- } else {
580
- this.debugLog('is not listening to Platform BLE')
581
- }
582
- }
583
-
584
- /**
585
- * Pushes the requested changes to the SwitchBot API
586
- */
587
- async pushChanges(): Promise<void> {
588
- if (this.BLE) {
589
- await this.BLEpushChanges()
590
- } else if (this.OpenAPI && this.platform.config.credentials?.token) {
591
- await this.openAPIpushChanges()
592
- } else {
593
- await this.offlineOff()
594
- this.debugWarnLog(`Connection Type: ${this.device.connectionType}, pushChanges will not happen.`)
595
- }
596
- // Refresh the status from the API
597
- interval(15000)
598
- .pipe(skipWhile(() => this.curtainUpdateInProgress))
599
- .pipe(take(1))
600
- .subscribe(async () => {
601
- await this.refreshStatus()
602
- })
603
- }
604
-
605
- async BLEpushChanges(): Promise<void> {
606
- this.debugLog('BLEpushChanges')
607
- if (this.WindowCovering.TargetPosition !== this.WindowCovering.CurrentPosition) {
608
- const switchBotBLE = await this.platform.connectBLE(this.accessory, this.device)
609
- try {
610
- const formattedDeviceId = formatDeviceIdAsMac(this.device.deviceId)
611
- this.device.bleMac = formattedDeviceId
612
- this.debugLog(`bleMac: ${this.device.bleMac}`)
613
- const { setPositionMode, Mode }: { setPositionMode: number, Mode: string } = await this.setPerformance()
614
- const adjustedMode = setPositionMode === 1 ? 0x01 : 0xFF
615
- this.debugLog(`Mode: ${Mode}, setPositionMode: ${setPositionMode}`)
616
- if (switchBotBLE !== false) {
617
- switchBotBLE
618
- .discover({ model: this.device.bleModel, quick: true, id: this.device.bleMac })
619
- .then(async (device_list: SwitchbotDevice[]) => {
620
- const deviceList = device_list as WoCurtain[]
621
- return await this.retryBLE({
622
- max: this.maxRetryBLE(),
623
- fn: async () => {
624
- if (deviceList && Array.isArray(deviceList) && deviceList.length > 0) {
625
- return await deviceList[0].runToPos(100 - Number(this.WindowCovering.TargetPosition), adjustedMode)
626
- } else {
627
- throw new Error('No device found')
628
- }
629
- },
630
- })
631
- })
632
- .then(async () => {
633
- this.successLog(`TargetPostion: ${this.WindowCovering.TargetPosition} sent over SwitchBot BLE, sent successfully`)
634
- await this.updateHomeKitCharacteristics()
635
- })
636
- .catch(async (e: any) => {
637
- await this.apiError(e)
638
- this.errorLog(`failed BLEpushChanges with ${this.device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`)
639
- await this.BLEPushConnection()
640
- })
641
- } else {
642
- this.errorLog(`wasn't able to establish BLE Connection, node-switchbot: ${JSON.stringify(switchBotBLE)}`)
643
- await this.BLEPushConnection()
644
- }
645
- } catch (error) {
646
- this.errorLog(`failed to format device ID as MAC, Error: ${error}`)
647
- }
648
- } else {
649
- this.debugLog(`No changes (BLEpushChanges), TargetPosition: ${this.WindowCovering.TargetPosition}, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
650
- }
651
- }
652
-
653
- async openAPIpushChanges(): Promise<void> {
654
- this.debugLog('openAPIpushChanges')
655
- if (this.WindowCovering.TargetPosition !== this.WindowCovering.CurrentPosition || this.device.disableCaching) {
656
- this.debugLog(`Pushing ${this.WindowCovering.TargetPosition}`)
657
- const adjustedTargetPosition = 100 - Number(this.WindowCovering.TargetPosition)
658
- const { setPositionMode, Mode }: { setPositionMode: number, Mode: string } = await this.setPerformance()
659
- this.debugLog(`Mode: ${Mode}, setPositionMode: ${setPositionMode}`)
660
- const adjustedMode = setPositionMode || 'ff'
661
- let bodyChange: bodyChange
662
- if (this.WindowCovering.HoldPosition) {
663
- bodyChange = {
664
- command: 'pause',
665
- parameter: 'default',
666
- commandType: 'command',
667
- }
668
- } else {
669
- bodyChange = {
670
- command: 'setPosition',
671
- parameter: `0,${adjustedMode},${adjustedTargetPosition}`,
672
- commandType: 'command',
673
- }
674
- }
675
- this.debugLog(`SwitchBot OpenAPI bodyChange: ${JSON.stringify(bodyChange)}`)
676
- try {
677
- const deviceStatus = await this.pushChangeRequest(bodyChange)
678
- this.debugLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
679
- if (await this.successfulStatusCodes(deviceStatus)) {
680
- this.debugSuccessLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
681
- await this.updateHomeKitCharacteristics()
682
- } else {
683
- await this.statusCode(deviceStatus.statusCode)
684
- }
685
- } catch (e: any) {
686
- await this.apiError(e)
687
- this.errorLog(`failed openAPIpushChanges with ${this.device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`)
688
- }
689
- } else {
690
- this.debugLog(`No changes (openAPIpushChanges), CurrentPosition: ${this.WindowCovering.CurrentPosition}, TargetPosition: ${this.WindowCovering.TargetPosition}`)
691
- }
692
- }
693
-
694
- /**
695
- * Handle requests to set the value of the "Target Position" characteristic
696
- */
697
- async TargetPositionSet(value: CharacteristicValue): Promise<void> {
698
- if (this.WindowCovering.TargetPosition !== this.accessory.context.TargetPosition) {
699
- this.infoLog(`Set TargetPosition: ${value}`)
700
- } else {
701
- this.debugLog(`No Changes, TargetPosition: ${value}`)
702
- }
703
-
704
- // Set HoldPosition to false when TargetPosition is changed
705
- this.WindowCovering.HoldPosition = false
706
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.HoldPosition, this.WindowCovering.HoldPosition)
707
-
708
- this.WindowCovering.TargetPosition = value
709
- await this.mqtt('TargetPosition', this.WindowCovering.TargetPosition)
710
- await this.mqtt('HoldPosition', this.WindowCovering.HoldPosition)
711
- await this.startUpdatingCurtainIfNeeded()
712
- }
713
-
714
- async startUpdatingCurtainIfNeeded() {
715
- await this.setMinMax()
716
- if (this.WindowCovering.TargetPosition > this.WindowCovering.CurrentPosition) {
717
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.INCREASING
718
- this.setNewTarget = true
719
- this.debugLog(`value: ${this.WindowCovering.TargetPosition}, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
720
- } else if (this.WindowCovering.TargetPosition < this.WindowCovering.CurrentPosition) {
721
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.DECREASING
722
- this.setNewTarget = true
723
- this.debugLog(`value: ${this.WindowCovering.TargetPosition}, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
724
- } else {
725
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.STOPPED
726
- this.setNewTarget = false
727
- this.debugLog(`value: ${this.WindowCovering.TargetPosition}, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
728
- }
729
- this.WindowCovering.Service.setCharacteristic(this.hap.Characteristic.PositionState, this.WindowCovering.PositionState)
730
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.PositionState).updateValue(this.WindowCovering.PositionState)
731
-
732
- /**
733
- * If Curtain movement time is short, the moving flag from backend is always false.
734
- * The minimum time depends on the network control latency.
735
- */
736
- clearTimeout(this.setNewTargetTimer)
737
- this.debugLog(`deviceUpdateRate: ${this.deviceUpdateRate}`)
738
- if (this.setNewTarget) {
739
- this.setNewTargetTimer = setTimeout(async () => {
740
- this.debugLog(`setNewTarget ${this.setNewTarget} timeout`)
741
- this.setNewTarget = false
742
- }, this.deviceUpdateRate * 1000)
743
- }
744
- this.doCurtainUpdate.next()
745
- }
746
-
747
- /**
748
- * Handle requests to set the value of the "Target Position" characteristic
749
- */
750
- async HoldPositionSet(value: CharacteristicValue): Promise<void> {
751
- this.debugLog(`HoldPosition: ${value}`)
752
- this.WindowCovering.HoldPosition = value
753
- this.doCurtainUpdate.next()
754
- }
755
-
756
- /**
757
- * Handle requests to set the value of the "Target Position" characteristic
758
- */
759
- async OpenModeSwitchSet(value: CharacteristicValue): Promise<void> {
760
- if (this.OpenModeSwitch && (this.device as curtainConfig).silentModeSwitch) {
761
- this.debugLog(`Silent Open Mode: ${value}`)
762
- this.OpenModeSwitch.On = value
763
- this.accessory.context.OpenModeSwitch.On = value
764
- if (value === true) {
765
- this.infoLog('Silent Open Mode is enabled')
766
- }
767
- }
768
- }
769
-
770
- /**
771
- * Handle requests to set the value of the "Target Position" characteristic
772
- */
773
- async CloseModeSwitchSet(value: CharacteristicValue): Promise<void> {
774
- if (this.CloseModeSwitch && (this.device as curtainConfig).silentModeSwitch) {
775
- this.debugLog(`Silent Close Mode: ${value}`)
776
- this.CloseModeSwitch.On = value
777
- this.accessory.context.CloseModeSwitch.On = value
778
- if (value === true) {
779
- this.infoLog('Silent Close Mode is enabled')
780
- }
781
- }
782
- }
783
-
784
- async updateHomeKitCharacteristics(): Promise<void> {
785
- await this.setMinMax()
786
-
787
- // Validate position values to prevent NaN
788
- if (Number.isNaN(Number(this.WindowCovering.CurrentPosition))) {
789
- this.warnLog(`CurrentPosition is NaN, keeping previous value`)
790
- this.WindowCovering.CurrentPosition = this.accessory.context.CurrentPosition ?? 100
791
- }
792
- if (Number.isNaN(Number(this.WindowCovering.TargetPosition))) {
793
- this.warnLog(`TargetPosition is NaN, keeping previous value`)
794
- this.WindowCovering.TargetPosition = this.accessory.context.TargetPosition ?? 100
795
- }
796
-
797
- // CurrentPosition
798
- await this.updateCharacteristic(this.WindowCovering.Service, this.hap.Characteristic.CurrentPosition, this.WindowCovering.CurrentPosition, 'CurrentPosition')
799
- // PositionState
800
- await this.updateCharacteristic(this.WindowCovering.Service, this.hap.Characteristic.PositionState, this.WindowCovering.PositionState, 'PositionState')
801
- // TargetPosition
802
- await this.updateCharacteristic(this.WindowCovering.Service, this.hap.Characteristic.TargetPosition, this.WindowCovering.TargetPosition, 'TargetPosition')
803
- // HoldPosition
804
- await this.updateCharacteristic(this.WindowCovering.Service, this.hap.Characteristic.HoldPosition, this.WindowCovering.HoldPosition, 'HoldPosition')
805
- // CurrentAmbientLightLevel
806
- if (!(this.device as curtainConfig).hide_lightsensor && this.LightSensor?.Service) {
807
- const history = { time: Math.round(new Date().valueOf() / 1000), lux: this.LightSensor.CurrentAmbientLightLevel }
808
- await this.updateCharacteristic(this.LightSensor?.Service, this.hap.Characteristic.CurrentAmbientLightLevel, this.LightSensor?.CurrentAmbientLightLevel, 'CurrentAmbientLightLevel', history)
809
- }
810
- // BatteryLevel
811
- await this.updateCharacteristic(this.Battery.Service, this.hap.Characteristic.BatteryLevel, this.Battery.BatteryLevel, 'BatteryLevel')
812
- // StatusLowBattery
813
- await this.updateCharacteristic(this.Battery.Service, this.hap.Characteristic.StatusLowBattery, this.Battery.StatusLowBattery, 'StatusLowBattery')
814
- // ChargingState
815
- await this.updateCharacteristic(this.Battery.Service, this.hap.Characteristic.ChargingState, this.Battery.ChargingState, 'ChargingState')
816
- }
817
-
818
- async BLEPushConnection() {
819
- this.warnLog('BLE connection failed for push operation')
820
- if (this.platform.config.credentials?.token && this.device.connectionType === 'BLE/OpenAPI') {
821
- this.warnLog('Using OpenAPI Connection to Push Changes')
822
- await this.openAPIpushChanges()
823
- } else {
824
- this.errorLog('No fallback available for BLE push failure. Consider using BLE/OpenAPI connection type.')
825
- }
826
- }
827
-
828
- async BLERefreshConnection(switchbot: SwitchBotBLE): Promise<void> {
829
- this.warnLog(`BLE connection failed for refresh operation, node-switchbot: ${switchbot}`)
830
- if (this.platform.config.credentials?.token && this.device.connectionType === 'BLE/OpenAPI') {
831
- this.warnLog('Using OpenAPI Connection to Refresh Status')
832
- await this.openAPIRefreshStatus()
833
- } else {
834
- this.errorLog('No fallback available for BLE refresh failure. Consider using BLE/OpenAPI connection type.')
835
- }
836
- }
837
-
838
- async setPerformance() {
839
- let setPositionMode: number
840
- let Mode: string
841
- if (Number(this.WindowCovering.TargetPosition) > 50) {
842
- if ((this.device as curtainConfig).setOpenMode === '1' || this.OpenModeSwitch?.On) {
843
- setPositionMode = 1
844
- Mode = 'Silent Mode'
845
- } else if ((this.device as curtainConfig).setOpenMode === '0' || !this.OpenModeSwitch?.On) {
846
- setPositionMode = 0
847
- Mode = 'Performance Mode'
848
- } else {
849
- setPositionMode = 0
850
- Mode = 'Default Mode'
851
- }
852
- } else {
853
- if ((this.device as curtainConfig).setCloseMode === '1' || this.CloseModeSwitch?.On) {
854
- setPositionMode = 1
855
- Mode = 'Silent Mode'
856
- } else if ((this.device as curtainConfig).setCloseMode === '0' || !this.CloseModeSwitch?.On) {
857
- setPositionMode = 0
858
- Mode = 'Performance Mode'
859
- } else {
860
- setPositionMode = 0
861
- Mode = 'Default Mode'
862
- }
863
- }
864
- this.infoLog(`Position Mode: ${setPositionMode}, Mode: ${Mode}`)
865
- return { setPositionMode, Mode }
866
- }
867
-
868
- async getCurrentPostion(): Promise<void> {
869
- await this.setMinMax()
870
- this.debugLog(`CurrentPosition ${this.WindowCovering.CurrentPosition}`)
871
- this.hasLoggedStandby = this.hasLoggedStandby ?? false
872
- if (this.deviceStatus ? (this.setNewTarget || this.deviceStatus.moving) : this.setNewTarget) {
873
- this.hasLoggedStandby = false
874
- this.infoLog('Checking Status ...')
875
- this.curtainMoving = true
876
- if (this.WindowCovering.TargetPosition > this.WindowCovering.CurrentPosition) {
877
- this.debugLog(`Closing, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
878
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.INCREASING
879
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.PositionState).updateValue(this.WindowCovering.PositionState)
880
- this.debugLog(`Increasing, PositionState: ${this.WindowCovering.PositionState}`)
881
- } else if (this.WindowCovering.TargetPosition < this.WindowCovering.CurrentPosition) {
882
- this.debugLog(`Opening, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
883
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.DECREASING
884
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.PositionState).updateValue(this.WindowCovering.PositionState)
885
- this.debugLog(`Decreasing, PositionState: ${this.WindowCovering.PositionState}`)
886
- } else {
887
- this.debugLog(`Standby, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
888
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.STOPPED
889
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.PositionState).updateValue(this.WindowCovering.PositionState)
890
- this.debugLog(`Stopped, PositionState: ${this.WindowCovering.PositionState}`)
891
- }
892
- } else {
893
- if (!this.hasLoggedStandby) {
894
- this.infoLog('Standby ...')
895
- this.hasLoggedStandby = true
896
- }
897
- this.curtainMoving = false
898
- this.debugLog(`Standby, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
899
- this.WindowCovering.TargetPosition = this.WindowCovering.CurrentPosition
900
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.STOPPED
901
- this.debugLog(`Stopped, PositionState: ${this.WindowCovering.PositionState}`)
902
- }
903
- this.debugLog(`CurrentPosition: ${this.WindowCovering.CurrentPosition}, TargetPosition: ${this.WindowCovering.TargetPosition}, PositionState: ${this.WindowCovering.PositionState},`)
904
- }
905
-
906
- async setMinMax(): Promise<void> {
907
- if ((this.device as curtainConfig).set_min) {
908
- if (Number(this.WindowCovering.CurrentPosition) >= (this.device as curtainConfig).set_min!) {
909
- this.WindowCovering.CurrentPosition = 100
910
- }
911
- }
912
- if ((this.device as curtainConfig).set_max) {
913
- if (Number(this.WindowCovering.CurrentPosition) <= (this.device as curtainConfig).set_max!) {
914
- this.WindowCovering.CurrentPosition = 0
915
- }
916
- }
917
- if (this.device.history) {
918
- const motion = this.accessory.getService(this.hap.Service.MotionSensor)
919
- const state = Number(this.WindowCovering.CurrentPosition) > 0 ? 1 : 0
920
- motion?.updateCharacteristic(this.hap.Characteristic.MotionDetected, state)
921
- }
922
- }
923
-
924
- async offlineOff(): Promise<void> {
925
- if (this.device.offline) {
926
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.CurrentPosition, 100)
927
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.TargetPosition, 100)
928
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.PositionState, this.hap.Characteristic.PositionState.STOPPED)
929
- }
930
- }
931
-
932
- async apiError(e: any): Promise<void> {
933
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.CurrentPosition, e)
934
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.PositionState, e)
935
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.TargetPosition, e)
936
- this.Battery.Service.updateCharacteristic(this.hap.Characteristic.BatteryLevel, e)
937
- this.Battery.Service.updateCharacteristic(this.hap.Characteristic.StatusLowBattery, e)
938
- this.Battery.Service.updateCharacteristic(this.hap.Characteristic.ChargingState, e)
939
- if (!(this.device as curtainConfig).hide_lightsensor && this.LightSensor?.Service) {
940
- this.LightSensor.Service.updateCharacteristic(this.hap.Characteristic.CurrentAmbientLightLevel, e)
941
- this.LightSensor.Service.updateCharacteristic(this.hap.Characteristic.StatusActive, e)
942
- }
943
- }
944
- }