@switchbot/homebridge-switchbot 5.0.0-beta.70 → 5.0.0-beta.72

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 (534) hide show
  1. package/.github/ISSUE_TEMPLATE/e2e-verification.md +36 -0
  2. package/.github/workflows/ci.yml +61 -0
  3. package/.github/workflows/manual-e2e.yml +103 -0
  4. package/CHANGELOG.md +20 -0
  5. package/E2E-VERIFICATION.md +121 -0
  6. package/MIGRATION.md +44 -0
  7. package/README.md +11 -0
  8. package/config.schema.json +99 -1940
  9. package/dist/deviceFactory.d.ts +13 -0
  10. package/dist/deviceFactory.d.ts.map +1 -0
  11. package/dist/deviceFactory.js +81 -0
  12. package/dist/deviceFactory.js.map +1 -0
  13. package/dist/devices/deviceBase.d.ts +50 -0
  14. package/dist/devices/deviceBase.d.ts.map +1 -0
  15. package/dist/devices/deviceBase.js +119 -0
  16. package/dist/devices/deviceBase.js.map +1 -0
  17. package/dist/devices/genericDevice.d.ts +283 -0
  18. package/dist/devices/genericDevice.d.ts.map +1 -0
  19. package/dist/devices/genericDevice.js +1035 -0
  20. package/dist/devices/genericDevice.js.map +1 -0
  21. package/dist/homebridge-ui/public/index.html +72 -440
  22. package/dist/homebridge-ui/server.d.ts +3 -1
  23. package/dist/homebridge-ui/server.d.ts.map +1 -1
  24. package/dist/homebridge-ui/server.js +47 -10
  25. package/dist/homebridge-ui/server.js.map +1 -1
  26. package/dist/index.d.ts.map +1 -1
  27. package/dist/index.js +1 -3
  28. package/dist/index.js.map +1 -1
  29. package/dist/platform.d.ts +27 -0
  30. package/dist/platform.d.ts.map +1 -0
  31. package/dist/platform.js +404 -0
  32. package/dist/platform.js.map +1 -0
  33. package/dist/settings.d.ts +10 -317
  34. package/dist/settings.d.ts.map +1 -1
  35. package/dist/settings.js +5 -30
  36. package/dist/settings.js.map +1 -1
  37. package/dist/switchbotClient.d.ts +32 -0
  38. package/dist/switchbotClient.d.ts.map +1 -0
  39. package/dist/switchbotClient.js +259 -0
  40. package/dist/switchbotClient.js.map +1 -0
  41. package/dist/utils.d.ts +36 -248
  42. package/dist/utils.d.ts.map +1 -1
  43. package/dist/utils.js +38 -1367
  44. package/dist/utils.js.map +1 -1
  45. package/docs/assets/icons.js +1 -1
  46. package/docs/assets/icons.svg +1 -1
  47. package/docs/assets/style.css +3 -3
  48. package/docs/index.html +50 -15
  49. package/docs/variables/default.html +1 -1
  50. package/package.json +19 -18
  51. package/scripts/e2e/README.md +25 -0
  52. package/scripts/e2e/curtain-e2e.sh +70 -0
  53. package/scripts/e2e/fan-e2e.sh +75 -0
  54. package/scripts/e2e/light-advanced-e2e.sh +97 -0
  55. package/scripts/e2e/light-e2e.sh +75 -0
  56. package/scripts/e2e/list-accessories.sh +19 -0
  57. package/scripts/e2e/lock-e2e.sh +65 -0
  58. package/scripts/generate-matter-maps.js +60 -0
  59. package/scripts/run-e2e-local.sh +14 -0
  60. package/src/deviceFactory.ts +122 -0
  61. package/src/devices/deviceBase.ts +141 -0
  62. package/src/devices/genericDevice.ts +965 -0
  63. package/src/homebridge-ui/public/index.html +72 -440
  64. package/src/homebridge-ui/server.ts +52 -10
  65. package/src/index.ts +1 -3
  66. package/src/platform.ts +395 -0
  67. package/src/settings.ts +12 -352
  68. package/src/switchbotClient.ts +266 -0
  69. package/src/utils.ts +47 -1456
  70. package/test/accessory-restore.spec.ts +73 -0
  71. package/test/device-mapping.spec.ts +37 -0
  72. package/test/deviceFactory.spec.ts +18 -0
  73. package/test/e2e/run-e2e.spec.ts +50 -0
  74. package/test/fan-swing.spec.ts +29 -0
  75. package/test/helpers/matter-harness.ts +53 -0
  76. package/test/lock-users.spec.ts +44 -0
  77. package/test/matter-childbridge.spec.ts +55 -0
  78. package/test/matter-descriptors.spec.ts +97 -0
  79. package/test/matter-device-state.spec.ts +101 -0
  80. package/test/matter-integration.spec.ts +70 -0
  81. package/test/platform.integration.spec.ts +55 -0
  82. package/test/switchbot-client-debounce.spec.ts +131 -0
  83. package/test/switchbot-client-openapi.spec.ts +56 -0
  84. package/test/switchbotClient.spec.ts +10 -0
  85. package/test/utils.spec.ts +20 -0
  86. package/vitest.config.ts +7 -0
  87. package/coverage/base.css +0 -224
  88. package/coverage/block-navigation.js +0 -87
  89. package/coverage/clover.xml +0 -15847
  90. package/coverage/coverage-final.json +0 -42
  91. package/coverage/docs/assets/dmt/dmt-component-data.js.html +0 -85
  92. package/coverage/docs/assets/dmt/dmt-components.js.html +0 -286
  93. package/coverage/docs/assets/dmt/index.html +0 -131
  94. package/coverage/docs/assets/hierarchy.js.html +0 -85
  95. package/coverage/docs/assets/icons.js.html +0 -136
  96. package/coverage/docs/assets/index.html +0 -146
  97. package/coverage/docs/assets/main.js.html +0 -265
  98. package/coverage/favicon.png +0 -0
  99. package/coverage/index.html +0 -191
  100. package/coverage/prettify.css +0 -1
  101. package/coverage/prettify.js +0 -2
  102. package/coverage/sort-arrow-sprite.png +0 -0
  103. package/coverage/sorter.js +0 -196
  104. package/coverage/src/device/blindtilt.ts.html +0 -3238
  105. package/coverage/src/device/bot.ts.html +0 -2803
  106. package/coverage/src/device/ceilinglight.ts.html +0 -2338
  107. package/coverage/src/device/colorbulb.ts.html +0 -2824
  108. package/coverage/src/device/contact.ts.html +0 -1465
  109. package/coverage/src/device/curtain.ts.html +0 -2869
  110. package/coverage/src/device/device.ts.html +0 -2500
  111. package/coverage/src/device/fan.ts.html +0 -2242
  112. package/coverage/src/device/hub.ts.html +0 -1408
  113. package/coverage/src/device/humidifier.ts.html +0 -2116
  114. package/coverage/src/device/index.html +0 -416
  115. package/coverage/src/device/iosensor.ts.html +0 -1375
  116. package/coverage/src/device/lightstrip.ts.html +0 -2617
  117. package/coverage/src/device/lock.ts.html +0 -1963
  118. package/coverage/src/device/meter.ts.html +0 -1372
  119. package/coverage/src/device/meterplus.ts.html +0 -1384
  120. package/coverage/src/device/meterpro.ts.html +0 -1618
  121. package/coverage/src/device/motion.ts.html +0 -1264
  122. package/coverage/src/device/plug.ts.html +0 -1372
  123. package/coverage/src/device/relayswitch.ts.html +0 -2284
  124. package/coverage/src/device/robotvacuumcleaner.ts.html +0 -1810
  125. package/coverage/src/device/waterdetector.ts.html +0 -1294
  126. package/coverage/src/homebridge-ui/index.html +0 -116
  127. package/coverage/src/homebridge-ui/server.ts.html +0 -229
  128. package/coverage/src/index.html +0 -161
  129. package/coverage/src/index.ts.html +0 -124
  130. package/coverage/src/irdevice/airconditioner.ts.html +0 -1687
  131. package/coverage/src/irdevice/airpurifier.ts.html +0 -844
  132. package/coverage/src/irdevice/camera.ts.html +0 -475
  133. package/coverage/src/irdevice/fan.ts.html +0 -766
  134. package/coverage/src/irdevice/index.html +0 -251
  135. package/coverage/src/irdevice/irdevice.ts.html +0 -1117
  136. package/coverage/src/irdevice/light.ts.html +0 -826
  137. package/coverage/src/irdevice/other.ts.html +0 -2458
  138. package/coverage/src/irdevice/tv.ts.html +0 -1222
  139. package/coverage/src/irdevice/vacuumcleaner.ts.html +0 -466
  140. package/coverage/src/irdevice/waterheater.ts.html +0 -469
  141. package/coverage/src/platform.ts.html +0 -8776
  142. package/coverage/src/settings.ts.html +0 -934
  143. package/coverage/src/utils.ts.html +0 -2092
  144. package/dist/devices-hap/airpurifier.d.ts +0 -54
  145. package/dist/devices-hap/airpurifier.d.ts.map +0 -1
  146. package/dist/devices-hap/airpurifier.js +0 -533
  147. package/dist/devices-hap/airpurifier.js.map +0 -1
  148. package/dist/devices-hap/blindtilt.d.ts +0 -90
  149. package/dist/devices-hap/blindtilt.d.ts.map +0 -1
  150. package/dist/devices-hap/blindtilt.js +0 -974
  151. package/dist/devices-hap/blindtilt.js.map +0 -1
  152. package/dist/devices-hap/bot.d.ts +0 -102
  153. package/dist/devices-hap/bot.d.ts.map +0 -1
  154. package/dist/devices-hap/bot.js +0 -822
  155. package/dist/devices-hap/bot.js.map +0 -1
  156. package/dist/devices-hap/ceilinglight.d.ts +0 -85
  157. package/dist/devices-hap/ceilinglight.d.ts.map +0 -1
  158. package/dist/devices-hap/ceilinglight.js +0 -707
  159. package/dist/devices-hap/ceilinglight.js.map +0 -1
  160. package/dist/devices-hap/colorbulb.d.ts +0 -88
  161. package/dist/devices-hap/colorbulb.d.ts.map +0 -1
  162. package/dist/devices-hap/colorbulb.js +0 -921
  163. package/dist/devices-hap/colorbulb.js.map +0 -1
  164. package/dist/devices-hap/contact.d.ts +0 -44
  165. package/dist/devices-hap/contact.d.ts.map +0 -1
  166. package/dist/devices-hap/contact.js +0 -409
  167. package/dist/devices-hap/contact.js.map +0 -1
  168. package/dist/devices-hap/curtain.d.ts +0 -73
  169. package/dist/devices-hap/curtain.d.ts.map +0 -1
  170. package/dist/devices-hap/curtain.js +0 -869
  171. package/dist/devices-hap/curtain.js.map +0 -1
  172. package/dist/devices-hap/device.d.ts +0 -108
  173. package/dist/devices-hap/device.d.ts.map +0 -1
  174. package/dist/devices-hap/device.js +0 -821
  175. package/dist/devices-hap/device.js.map +0 -1
  176. package/dist/devices-hap/fan.d.ts +0 -69
  177. package/dist/devices-hap/fan.d.ts.map +0 -1
  178. package/dist/devices-hap/fan.js +0 -655
  179. package/dist/devices-hap/fan.js.map +0 -1
  180. package/dist/devices-hap/hub.d.ts +0 -37
  181. package/dist/devices-hap/hub.d.ts.map +0 -1
  182. package/dist/devices-hap/hub.js +0 -393
  183. package/dist/devices-hap/hub.js.map +0 -1
  184. package/dist/devices-hap/humidifier.d.ts +0 -73
  185. package/dist/devices-hap/humidifier.d.ts.map +0 -1
  186. package/dist/devices-hap/humidifier.js +0 -716
  187. package/dist/devices-hap/humidifier.js.map +0 -1
  188. package/dist/devices-hap/iosensor.d.ts +0 -42
  189. package/dist/devices-hap/iosensor.d.ts.map +0 -1
  190. package/dist/devices-hap/iosensor.js +0 -397
  191. package/dist/devices-hap/iosensor.js.map +0 -1
  192. package/dist/devices-hap/lightstrip.d.ts +0 -79
  193. package/dist/devices-hap/lightstrip.d.ts.map +0 -1
  194. package/dist/devices-hap/lightstrip.js +0 -827
  195. package/dist/devices-hap/lightstrip.js.map +0 -1
  196. package/dist/devices-hap/lock.d.ts +0 -53
  197. package/dist/devices-hap/lock.d.ts.map +0 -1
  198. package/dist/devices-hap/lock.js +0 -569
  199. package/dist/devices-hap/lock.js.map +0 -1
  200. package/dist/devices-hap/meter.d.ts +0 -37
  201. package/dist/devices-hap/meter.d.ts.map +0 -1
  202. package/dist/devices-hap/meter.js +0 -380
  203. package/dist/devices-hap/meter.js.map +0 -1
  204. package/dist/devices-hap/meterplus.d.ts +0 -42
  205. package/dist/devices-hap/meterplus.d.ts.map +0 -1
  206. package/dist/devices-hap/meterplus.js +0 -385
  207. package/dist/devices-hap/meterplus.js.map +0 -1
  208. package/dist/devices-hap/meterpro.d.ts +0 -43
  209. package/dist/devices-hap/meterpro.d.ts.map +0 -1
  210. package/dist/devices-hap/meterpro.js +0 -469
  211. package/dist/devices-hap/meterpro.js.map +0 -1
  212. package/dist/devices-hap/motion.d.ts +0 -42
  213. package/dist/devices-hap/motion.d.ts.map +0 -1
  214. package/dist/devices-hap/motion.js +0 -345
  215. package/dist/devices-hap/motion.js.map +0 -1
  216. package/dist/devices-hap/plug.d.ts +0 -49
  217. package/dist/devices-hap/plug.d.ts.map +0 -1
  218. package/dist/devices-hap/plug.js +0 -400
  219. package/dist/devices-hap/plug.js.map +0 -1
  220. package/dist/devices-hap/relayswitch.d.ts +0 -96
  221. package/dist/devices-hap/relayswitch.d.ts.map +0 -1
  222. package/dist/devices-hap/relayswitch.js +0 -642
  223. package/dist/devices-hap/relayswitch.js.map +0 -1
  224. package/dist/devices-hap/robotvacuumcleaner.d.ts +0 -54
  225. package/dist/devices-hap/robotvacuumcleaner.d.ts.map +0 -1
  226. package/dist/devices-hap/robotvacuumcleaner.js +0 -530
  227. package/dist/devices-hap/robotvacuumcleaner.js.map +0 -1
  228. package/dist/devices-hap/waterdetector.d.ts +0 -41
  229. package/dist/devices-hap/waterdetector.d.ts.map +0 -1
  230. package/dist/devices-hap/waterdetector.js +0 -356
  231. package/dist/devices-hap/waterdetector.js.map +0 -1
  232. package/dist/devices-matter/BaseMatterAccessory.d.ts +0 -90
  233. package/dist/devices-matter/BaseMatterAccessory.d.ts.map +0 -1
  234. package/dist/devices-matter/BaseMatterAccessory.js +0 -264
  235. package/dist/devices-matter/BaseMatterAccessory.js.map +0 -1
  236. package/dist/devices-matter/ColorLightAccessory.d.ts +0 -20
  237. package/dist/devices-matter/ColorLightAccessory.d.ts.map +0 -1
  238. package/dist/devices-matter/ColorLightAccessory.js +0 -95
  239. package/dist/devices-matter/ColorLightAccessory.js.map +0 -1
  240. package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts +0 -18
  241. package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts.map +0 -1
  242. package/dist/devices-matter/ColorTemperatureLightAccessory.js +0 -76
  243. package/dist/devices-matter/ColorTemperatureLightAccessory.js.map +0 -1
  244. package/dist/devices-matter/ContactSensorAccessory.d.ts +0 -12
  245. package/dist/devices-matter/ContactSensorAccessory.d.ts.map +0 -1
  246. package/dist/devices-matter/ContactSensorAccessory.js +0 -34
  247. package/dist/devices-matter/ContactSensorAccessory.js.map +0 -1
  248. package/dist/devices-matter/DimmableLightAccessory.d.ts +0 -58
  249. package/dist/devices-matter/DimmableLightAccessory.d.ts.map +0 -1
  250. package/dist/devices-matter/DimmableLightAccessory.js +0 -167
  251. package/dist/devices-matter/DimmableLightAccessory.js.map +0 -1
  252. package/dist/devices-matter/DoorLockAccessory.d.ts +0 -14
  253. package/dist/devices-matter/DoorLockAccessory.d.ts.map +0 -1
  254. package/dist/devices-matter/DoorLockAccessory.js +0 -50
  255. package/dist/devices-matter/DoorLockAccessory.js.map +0 -1
  256. package/dist/devices-matter/ExtendedColorLightAccessory.d.ts +0 -21
  257. package/dist/devices-matter/ExtendedColorLightAccessory.d.ts.map +0 -1
  258. package/dist/devices-matter/ExtendedColorLightAccessory.js +0 -106
  259. package/dist/devices-matter/ExtendedColorLightAccessory.js.map +0 -1
  260. package/dist/devices-matter/FanAccessory.d.ts +0 -16
  261. package/dist/devices-matter/FanAccessory.d.ts.map +0 -1
  262. package/dist/devices-matter/FanAccessory.js +0 -81
  263. package/dist/devices-matter/FanAccessory.js.map +0 -1
  264. package/dist/devices-matter/HumiditySensorAccessory.d.ts +0 -12
  265. package/dist/devices-matter/HumiditySensorAccessory.d.ts.map +0 -1
  266. package/dist/devices-matter/HumiditySensorAccessory.js +0 -34
  267. package/dist/devices-matter/HumiditySensorAccessory.js.map +0 -1
  268. package/dist/devices-matter/LeakSensorAccessory.d.ts +0 -12
  269. package/dist/devices-matter/LeakSensorAccessory.d.ts.map +0 -1
  270. package/dist/devices-matter/LeakSensorAccessory.js +0 -33
  271. package/dist/devices-matter/LeakSensorAccessory.js.map +0 -1
  272. package/dist/devices-matter/LightSensorAccessory.d.ts +0 -12
  273. package/dist/devices-matter/LightSensorAccessory.d.ts.map +0 -1
  274. package/dist/devices-matter/LightSensorAccessory.js +0 -34
  275. package/dist/devices-matter/LightSensorAccessory.js.map +0 -1
  276. package/dist/devices-matter/OccupancySensorAccessory.d.ts +0 -12
  277. package/dist/devices-matter/OccupancySensorAccessory.d.ts.map +0 -1
  278. package/dist/devices-matter/OccupancySensorAccessory.js +0 -39
  279. package/dist/devices-matter/OccupancySensorAccessory.js.map +0 -1
  280. package/dist/devices-matter/OnOffLightAccessory.d.ts +0 -38
  281. package/dist/devices-matter/OnOffLightAccessory.d.ts.map +0 -1
  282. package/dist/devices-matter/OnOffLightAccessory.js +0 -110
  283. package/dist/devices-matter/OnOffLightAccessory.js.map +0 -1
  284. package/dist/devices-matter/OnOffOutletAccessory.d.ts +0 -14
  285. package/dist/devices-matter/OnOffOutletAccessory.d.ts.map +0 -1
  286. package/dist/devices-matter/OnOffOutletAccessory.js +0 -43
  287. package/dist/devices-matter/OnOffOutletAccessory.js.map +0 -1
  288. package/dist/devices-matter/OnOffSwitchAccessory.d.ts +0 -14
  289. package/dist/devices-matter/OnOffSwitchAccessory.d.ts.map +0 -1
  290. package/dist/devices-matter/OnOffSwitchAccessory.js +0 -42
  291. package/dist/devices-matter/OnOffSwitchAccessory.js.map +0 -1
  292. package/dist/devices-matter/RoboticVacuumAccessory.d.ts +0 -61
  293. package/dist/devices-matter/RoboticVacuumAccessory.d.ts.map +0 -1
  294. package/dist/devices-matter/RoboticVacuumAccessory.js +0 -544
  295. package/dist/devices-matter/RoboticVacuumAccessory.js.map +0 -1
  296. package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts +0 -11
  297. package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts.map +0 -1
  298. package/dist/devices-matter/SmokeCOAlarmAccessory.js +0 -49
  299. package/dist/devices-matter/SmokeCOAlarmAccessory.js.map +0 -1
  300. package/dist/devices-matter/TemperatureSensorAccessory.d.ts +0 -12
  301. package/dist/devices-matter/TemperatureSensorAccessory.d.ts.map +0 -1
  302. package/dist/devices-matter/TemperatureSensorAccessory.js +0 -36
  303. package/dist/devices-matter/TemperatureSensorAccessory.js.map +0 -1
  304. package/dist/devices-matter/ThermostatAccessory.d.ts +0 -19
  305. package/dist/devices-matter/ThermostatAccessory.d.ts.map +0 -1
  306. package/dist/devices-matter/ThermostatAccessory.js +0 -95
  307. package/dist/devices-matter/ThermostatAccessory.js.map +0 -1
  308. package/dist/devices-matter/VenetianBlindAccessory.d.ts +0 -19
  309. package/dist/devices-matter/VenetianBlindAccessory.d.ts.map +0 -1
  310. package/dist/devices-matter/VenetianBlindAccessory.js +0 -99
  311. package/dist/devices-matter/VenetianBlindAccessory.js.map +0 -1
  312. package/dist/devices-matter/WindowBlindAccessory.d.ts +0 -17
  313. package/dist/devices-matter/WindowBlindAccessory.d.ts.map +0 -1
  314. package/dist/devices-matter/WindowBlindAccessory.js +0 -131
  315. package/dist/devices-matter/WindowBlindAccessory.js.map +0 -1
  316. package/dist/devices-matter/custom/PowerStripAccessory.d.ts +0 -97
  317. package/dist/devices-matter/custom/PowerStripAccessory.d.ts.map +0 -1
  318. package/dist/devices-matter/custom/PowerStripAccessory.js +0 -265
  319. package/dist/devices-matter/custom/PowerStripAccessory.js.map +0 -1
  320. package/dist/devices-matter/custom/index.d.ts +0 -8
  321. package/dist/devices-matter/custom/index.d.ts.map +0 -1
  322. package/dist/devices-matter/custom/index.js +0 -8
  323. package/dist/devices-matter/custom/index.js.map +0 -1
  324. package/dist/devices-matter/index.d.ts +0 -29
  325. package/dist/devices-matter/index.d.ts.map +0 -1
  326. package/dist/devices-matter/index.js +0 -28
  327. package/dist/devices-matter/index.js.map +0 -1
  328. package/dist/irdevice/airconditioner.d.ts +0 -61
  329. package/dist/irdevice/airconditioner.d.ts.map +0 -1
  330. package/dist/irdevice/airconditioner.js +0 -472
  331. package/dist/irdevice/airconditioner.js.map +0 -1
  332. package/dist/irdevice/airpurifier.d.ts +0 -50
  333. package/dist/irdevice/airpurifier.d.ts.map +0 -1
  334. package/dist/irdevice/airpurifier.js +0 -213
  335. package/dist/irdevice/airpurifier.js.map +0 -1
  336. package/dist/irdevice/camera.d.ts +0 -32
  337. package/dist/irdevice/camera.d.ts.map +0 -1
  338. package/dist/irdevice/camera.js +0 -107
  339. package/dist/irdevice/camera.js.map +0 -1
  340. package/dist/irdevice/fan.d.ts +0 -36
  341. package/dist/irdevice/fan.d.ts.map +0 -1
  342. package/dist/irdevice/fan.js +0 -200
  343. package/dist/irdevice/fan.js.map +0 -1
  344. package/dist/irdevice/irdevice.d.ts +0 -69
  345. package/dist/irdevice/irdevice.d.ts.map +0 -1
  346. package/dist/irdevice/irdevice.js +0 -339
  347. package/dist/irdevice/irdevice.js.map +0 -1
  348. package/dist/irdevice/light.d.ts +0 -36
  349. package/dist/irdevice/light.d.ts.map +0 -1
  350. package/dist/irdevice/light.js +0 -206
  351. package/dist/irdevice/light.js.map +0 -1
  352. package/dist/irdevice/other.d.ts +0 -57
  353. package/dist/irdevice/other.d.ts.map +0 -1
  354. package/dist/irdevice/other.js +0 -778
  355. package/dist/irdevice/other.js.map +0 -1
  356. package/dist/irdevice/tv.d.ts +0 -45
  357. package/dist/irdevice/tv.d.ts.map +0 -1
  358. package/dist/irdevice/tv.js +0 -327
  359. package/dist/irdevice/tv.js.map +0 -1
  360. package/dist/irdevice/vacuumcleaner.d.ts +0 -28
  361. package/dist/irdevice/vacuumcleaner.d.ts.map +0 -1
  362. package/dist/irdevice/vacuumcleaner.js +0 -104
  363. package/dist/irdevice/vacuumcleaner.js.map +0 -1
  364. package/dist/irdevice/waterheater.d.ts +0 -30
  365. package/dist/irdevice/waterheater.d.ts.map +0 -1
  366. package/dist/irdevice/waterheater.js +0 -105
  367. package/dist/irdevice/waterheater.js.map +0 -1
  368. package/dist/platform-hap.d.ts +0 -160
  369. package/dist/platform-hap.d.ts.map +0 -1
  370. package/dist/platform-hap.js +0 -3041
  371. package/dist/platform-hap.js.map +0 -1
  372. package/dist/platform-matter.d.ts +0 -188
  373. package/dist/platform-matter.d.ts.map +0 -1
  374. package/dist/platform-matter.js +0 -2545
  375. package/dist/platform-matter.js.map +0 -1
  376. package/dist/test/apiRequestTracker.test.d.ts +0 -2
  377. package/dist/test/apiRequestTracker.test.d.ts.map +0 -1
  378. package/dist/test/apiRequestTracker.test.js +0 -392
  379. package/dist/test/apiRequestTracker.test.js.map +0 -1
  380. package/dist/test/hap/device-webhook-context.test.d.ts +0 -2
  381. package/dist/test/hap/device-webhook-context.test.d.ts.map +0 -1
  382. package/dist/test/hap/device-webhook-context.test.js +0 -128
  383. package/dist/test/hap/device-webhook-context.test.js.map +0 -1
  384. package/dist/test/hap/platform-hap.logging.test.d.ts +0 -2
  385. package/dist/test/hap/platform-hap.logging.test.d.ts.map +0 -1
  386. package/dist/test/hap/platform-hap.logging.test.js +0 -33
  387. package/dist/test/hap/platform-hap.logging.test.js.map +0 -1
  388. package/dist/test/hap/platform-hap.test.d.ts +0 -2
  389. package/dist/test/hap/platform-hap.test.d.ts.map +0 -1
  390. package/dist/test/hap/platform-hap.test.js +0 -62
  391. package/dist/test/hap/platform-hap.test.js.map +0 -1
  392. package/dist/test/helpers/platform-fixtures.d.ts +0 -9
  393. package/dist/test/helpers/platform-fixtures.d.ts.map +0 -1
  394. package/dist/test/helpers/platform-fixtures.js +0 -30
  395. package/dist/test/helpers/platform-fixtures.js.map +0 -1
  396. package/dist/test/homebridge-ui/server.test.d.ts +0 -2
  397. package/dist/test/homebridge-ui/server.test.d.ts.map +0 -1
  398. package/dist/test/homebridge-ui/server.test.js +0 -445
  399. package/dist/test/homebridge-ui/server.test.js.map +0 -1
  400. package/dist/test/index.test.d.ts +0 -2
  401. package/dist/test/index.test.d.ts.map +0 -1
  402. package/dist/test/index.test.js +0 -19
  403. package/dist/test/index.test.js.map +0 -1
  404. package/dist/test/matter/devices-matter/baseMatterAccessory.test.d.ts +0 -2
  405. package/dist/test/matter/devices-matter/baseMatterAccessory.test.d.ts.map +0 -1
  406. package/dist/test/matter/devices-matter/baseMatterAccessory.test.js +0 -71
  407. package/dist/test/matter/devices-matter/baseMatterAccessory.test.js.map +0 -1
  408. package/dist/test/matter/platform-matter.additional.test.d.ts +0 -2
  409. package/dist/test/matter/platform-matter.additional.test.d.ts.map +0 -1
  410. package/dist/test/matter/platform-matter.additional.test.js +0 -35
  411. package/dist/test/matter/platform-matter.additional.test.js.map +0 -1
  412. package/dist/test/matter/platform-matter.bleparse.test.d.ts +0 -2
  413. package/dist/test/matter/platform-matter.bleparse.test.d.ts.map +0 -1
  414. package/dist/test/matter/platform-matter.bleparse.test.js +0 -43
  415. package/dist/test/matter/platform-matter.bleparse.test.js.map +0 -1
  416. package/dist/test/matter/platform-matter.cleanup.test.d.ts +0 -2
  417. package/dist/test/matter/platform-matter.cleanup.test.d.ts.map +0 -1
  418. package/dist/test/matter/platform-matter.cleanup.test.js +0 -70
  419. package/dist/test/matter/platform-matter.cleanup.test.js.map +0 -1
  420. package/dist/test/matter/platform-matter.keepstale.test.d.ts +0 -2
  421. package/dist/test/matter/platform-matter.keepstale.test.d.ts.map +0 -1
  422. package/dist/test/matter/platform-matter.keepstale.test.js +0 -27
  423. package/dist/test/matter/platform-matter.keepstale.test.js.map +0 -1
  424. package/dist/test/matter/platform-matter.logging.test.d.ts +0 -2
  425. package/dist/test/matter/platform-matter.logging.test.d.ts.map +0 -1
  426. package/dist/test/matter/platform-matter.logging.test.js +0 -29
  427. package/dist/test/matter/platform-matter.logging.test.js.map +0 -1
  428. package/dist/test/matter/platform-matter.mapping.test.d.ts +0 -2
  429. package/dist/test/matter/platform-matter.mapping.test.d.ts.map +0 -1
  430. package/dist/test/matter/platform-matter.mapping.test.js +0 -43
  431. package/dist/test/matter/platform-matter.mapping.test.js.map +0 -1
  432. package/dist/test/matter/platform-matter.openapi-mapping.test.d.ts +0 -2
  433. package/dist/test/matter/platform-matter.openapi-mapping.test.d.ts.map +0 -1
  434. package/dist/test/matter/platform-matter.openapi-mapping.test.js +0 -84
  435. package/dist/test/matter/platform-matter.openapi-mapping.test.js.map +0 -1
  436. package/dist/test/matter/platform-matter.test.d.ts +0 -2
  437. package/dist/test/matter/platform-matter.test.d.ts.map +0 -1
  438. package/dist/test/matter/platform-matter.test.js +0 -117
  439. package/dist/test/matter/platform-matter.test.js.map +0 -1
  440. package/dist/test/matter/platform-matter.unregister.test.d.ts +0 -2
  441. package/dist/test/matter/platform-matter.unregister.test.d.ts.map +0 -1
  442. package/dist/test/matter/platform-matter.unregister.test.js +0 -30
  443. package/dist/test/matter/platform-matter.unregister.test.js.map +0 -1
  444. package/dist/test/matter/platform-matter.webhook.test.d.ts +0 -2
  445. package/dist/test/matter/platform-matter.webhook.test.d.ts.map +0 -1
  446. package/dist/test/matter/platform-matter.webhook.test.js +0 -46
  447. package/dist/test/matter/platform-matter.webhook.test.js.map +0 -1
  448. package/dist/test/utils.test.d.ts +0 -2
  449. package/dist/test/utils.test.d.ts.map +0 -1
  450. package/dist/test/utils.test.js +0 -95
  451. package/dist/test/utils.test.js.map +0 -1
  452. package/dist/test/verifyconfig.test.d.ts +0 -2
  453. package/dist/test/verifyconfig.test.d.ts.map +0 -1
  454. package/dist/test/verifyconfig.test.js +0 -167
  455. package/dist/test/verifyconfig.test.js.map +0 -1
  456. package/src/custom.d.ts +0 -7
  457. package/src/devices-hap/airpurifier.ts +0 -568
  458. package/src/devices-hap/blindtilt.ts +0 -1049
  459. package/src/devices-hap/bot.ts +0 -910
  460. package/src/devices-hap/ceilinglight.ts +0 -747
  461. package/src/devices-hap/colorbulb.ts +0 -940
  462. package/src/devices-hap/contact.ts +0 -457
  463. package/src/devices-hap/curtain.ts +0 -944
  464. package/src/devices-hap/device.ts +0 -890
  465. package/src/devices-hap/fan.ts +0 -716
  466. package/src/devices-hap/hub.ts +0 -440
  467. package/src/devices-hap/humidifier.ts +0 -762
  468. package/src/devices-hap/iosensor.ts +0 -442
  469. package/src/devices-hap/lightstrip.ts +0 -863
  470. package/src/devices-hap/lock.ts +0 -627
  471. package/src/devices-hap/meter.ts +0 -427
  472. package/src/devices-hap/meterplus.ts +0 -431
  473. package/src/devices-hap/meterpro.ts +0 -523
  474. package/src/devices-hap/motion.ts +0 -390
  475. package/src/devices-hap/plug.ts +0 -427
  476. package/src/devices-hap/relayswitch.ts +0 -727
  477. package/src/devices-hap/robotvacuumcleaner.ts +0 -574
  478. package/src/devices-hap/waterdetector.ts +0 -400
  479. package/src/devices-matter/BaseMatterAccessory.ts +0 -302
  480. package/src/devices-matter/ColorLightAccessory.ts +0 -110
  481. package/src/devices-matter/ColorTemperatureLightAccessory.ts +0 -90
  482. package/src/devices-matter/ContactSensorAccessory.ts +0 -41
  483. package/src/devices-matter/DimmableLightAccessory.ts +0 -192
  484. package/src/devices-matter/DoorLockAccessory.ts +0 -60
  485. package/src/devices-matter/ExtendedColorLightAccessory.ts +0 -122
  486. package/src/devices-matter/FanAccessory.ts +0 -95
  487. package/src/devices-matter/HumiditySensorAccessory.ts +0 -41
  488. package/src/devices-matter/LeakSensorAccessory.ts +0 -40
  489. package/src/devices-matter/LightSensorAccessory.ts +0 -41
  490. package/src/devices-matter/OccupancySensorAccessory.ts +0 -48
  491. package/src/devices-matter/OnOffLightAccessory.ts +0 -125
  492. package/src/devices-matter/OnOffOutletAccessory.ts +0 -51
  493. package/src/devices-matter/OnOffSwitchAccessory.ts +0 -51
  494. package/src/devices-matter/RoboticVacuumAccessory.ts +0 -621
  495. package/src/devices-matter/SmokeCOAlarmAccessory.ts +0 -59
  496. package/src/devices-matter/TemperatureSensorAccessory.ts +0 -43
  497. package/src/devices-matter/ThermostatAccessory.ts +0 -110
  498. package/src/devices-matter/VenetianBlindAccessory.ts +0 -115
  499. package/src/devices-matter/WindowBlindAccessory.ts +0 -135
  500. package/src/devices-matter/custom/PowerStripAccessory.ts +0 -309
  501. package/src/devices-matter/custom/index.ts +0 -8
  502. package/src/devices-matter/index.ts +0 -29
  503. package/src/irdevice/airconditioner.ts +0 -533
  504. package/src/irdevice/airpurifier.ts +0 -252
  505. package/src/irdevice/camera.ts +0 -129
  506. package/src/irdevice/fan.ts +0 -226
  507. package/src/irdevice/irdevice.ts +0 -383
  508. package/src/irdevice/light.ts +0 -246
  509. package/src/irdevice/other.ts +0 -790
  510. package/src/irdevice/tv.ts +0 -378
  511. package/src/irdevice/vacuumcleaner.ts +0 -126
  512. package/src/irdevice/waterheater.ts +0 -127
  513. package/src/platform-hap.ts +0 -3193
  514. package/src/platform-matter.ts +0 -2703
  515. package/src/test/apiRequestTracker.test.ts +0 -417
  516. package/src/test/hap/device-webhook-context.test.ts +0 -136
  517. package/src/test/hap/platform-hap.logging.test.ts +0 -36
  518. package/src/test/hap/platform-hap.test.ts +0 -70
  519. package/src/test/helpers/platform-fixtures.ts +0 -33
  520. package/src/test/homebridge-ui/server.test.ts +0 -486
  521. package/src/test/index.test.ts +0 -24
  522. package/src/test/matter/devices-matter/baseMatterAccessory.test.ts +0 -88
  523. package/src/test/matter/platform-matter.additional.test.ts +0 -44
  524. package/src/test/matter/platform-matter.bleparse.test.ts +0 -47
  525. package/src/test/matter/platform-matter.cleanup.test.ts +0 -86
  526. package/src/test/matter/platform-matter.keepstale.test.ts +0 -37
  527. package/src/test/matter/platform-matter.logging.test.ts +0 -33
  528. package/src/test/matter/platform-matter.mapping.test.ts +0 -57
  529. package/src/test/matter/platform-matter.openapi-mapping.test.ts +0 -109
  530. package/src/test/matter/platform-matter.test.ts +0 -144
  531. package/src/test/matter/platform-matter.unregister.test.ts +0 -39
  532. package/src/test/matter/platform-matter.webhook.test.ts +0 -54
  533. package/src/test/utils.test.ts +0 -96
  534. package/src/test/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 === 1 ? '01' : '00'
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
- }