@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,1049 +0,0 @@
1
- /* Copyright(C) 2021-2024, donavanbecker (https://github.com/donavanbecker). All rights reserved.
2
- *
3
- * blindtilt.ts: @switchbot/homebridge-switchbot.
4
- */
5
- import type { CharacteristicValue, PlatformAccessory, Service } from 'homebridge'
6
- import type { blindTiltServiceData, blindTiltStatus, blindTiltWebhookContext, bodyChange, device, SwitchBotBLE, SwitchbotDevice, WoBlindTilt } from 'node-switchbot'
7
-
8
- import type { SwitchBotHAPPlatform } from '../platform-hap.js'
9
- import type { blindTiltConfig, devicesConfig } from '../settings.js'
10
-
11
- /*
12
- * For Testing Locally:
13
- * import { SwitchBotBLEModel, SwitchBotBLEModelName } from '/Users/Shared/GitHub/OpenWonderLabs/node-switchbot/dist/index.js';
14
- */
15
- import { SwitchBotBLEModel, SwitchBotBLEModelName } from 'node-switchbot'
16
- import { debounceTime, interval, skipWhile, Subject, take, tap } from 'rxjs'
17
-
18
- import { BlindTiltMappingMode, formatDeviceIdAsMac } from '../utils.js'
19
- import { deviceBase } from './device.js'
20
-
21
- export class BlindTilt extends deviceBase {
22
- // Services
23
- private WindowCovering: {
24
- Name: CharacteristicValue
25
- Service: Service
26
- PositionState: CharacteristicValue
27
- TargetPosition: CharacteristicValue
28
- CurrentPosition: CharacteristicValue
29
- TargetHorizontalTiltAngle: CharacteristicValue
30
- CurrentHorizontalTiltAngle: CharacteristicValue
31
- }
32
-
33
- private Battery: {
34
- Name: CharacteristicValue
35
- Service: Service
36
- BatteryLevel: CharacteristicValue
37
- StatusLowBattery: CharacteristicValue
38
- ChargingState?: CharacteristicValue
39
- }
40
-
41
- private LightSensor?: {
42
- Name: CharacteristicValue
43
- Service: Service
44
- CurrentAmbientLightLevel?: CharacteristicValue
45
- }
46
-
47
- private OpenModeSwitch?: {
48
- Name: CharacteristicValue
49
- Service: Service
50
- On: CharacteristicValue
51
- }
52
-
53
- private CloseModeSwitch?: {
54
- Name: CharacteristicValue
55
- Service: Service
56
- On: CharacteristicValue
57
- }
58
-
59
- // OpenAPI
60
- deviceStatus!: blindTiltStatus
61
- mappingMode: BlindTiltMappingMode = BlindTiltMappingMode.OnlyUp
62
-
63
- // Webhook
64
- webhookContext!: blindTiltWebhookContext
65
-
66
- // BLE
67
- serviceData!: blindTiltServiceData
68
-
69
- // Target
70
- setNewTarget!: boolean
71
- setNewTargetTimer!: NodeJS.Timeout
72
-
73
- // Updates
74
- blindTiltMoving: boolean
75
- blindTiltUpdateInProgress: boolean
76
- doBlindTiltUpdate: Subject<void>
77
-
78
- constructor(
79
- readonly platform: SwitchBotHAPPlatform,
80
- accessory: PlatformAccessory,
81
- device: device & devicesConfig,
82
- ) {
83
- super(platform, accessory, device)
84
- // Set category
85
- accessory.category = this.hap.Categories.WINDOW_COVERING
86
-
87
- // default placeholders
88
-
89
- this.mappingMode = ((device as blindTiltConfig).mapping as BlindTiltMappingMode) ?? BlindTiltMappingMode.OnlyUp
90
- this.debugLog(`Mapping mode: ${this.mappingMode}`)
91
-
92
- // this is subject we use to track when we need to POST changes to the SwitchBot API
93
- this.doBlindTiltUpdate = new Subject()
94
- this.blindTiltMoving = false
95
- this.blindTiltUpdateInProgress = false
96
- this.setNewTarget = false
97
-
98
- // Initialize WindowCovering Service
99
- accessory.context.WindowCovering = accessory.context.WindowCovering ?? {}
100
- this.WindowCovering = {
101
- Name: accessory.displayName,
102
- Service: accessory.getService(this.hap.Service.WindowCovering) ?? accessory.addService(this.hap.Service.WindowCovering) as Service,
103
- PositionState: accessory.context.PositionState ?? this.hap.Characteristic.PositionState.STOPPED,
104
- TargetPosition: accessory.context.TargetPosition ?? 100,
105
- CurrentPosition: accessory.context.CurrentPosition ?? 100,
106
- TargetHorizontalTiltAngle: accessory.context.TargetHorizontalTiltAngle ?? 90,
107
- CurrentHorizontalTiltAngle: accessory.context.CurrentHorizontalTiltAngle ?? 90,
108
- }
109
- accessory.context.WindowCovering = this.WindowCovering as object
110
-
111
- // Initialize WindowCovering Characteristics
112
- this.WindowCovering.Service.setCharacteristic(this.hap.Characteristic.Name, this.WindowCovering.Name).getCharacteristic(this.hap.Characteristic.TargetPosition).setProps({
113
- minStep: (device as blindTiltConfig).set_minStep ?? 1,
114
- minValue: 0,
115
- maxValue: 100,
116
- validValueRanges: [0, 100],
117
- }).onGet(() => {
118
- return this.WindowCovering.TargetPosition
119
- }).onSet(this.TargetPositionSet.bind(this))
120
-
121
- // Initialize WindowCovering CurrentPosition Characteristic
122
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.CurrentPosition).setProps({
123
- minStep: (device as blindTiltConfig).set_minStep ?? 1,
124
- minValue: 0,
125
- maxValue: 100,
126
- validValueRanges: [0, 100],
127
- }).onGet(() => {
128
- return this.WindowCovering.CurrentPosition ?? 0
129
- })
130
-
131
- // Initialize WindowCovering TargetHorizontalTiltAngle Characteristic
132
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.TargetHorizontalTiltAngle).setProps({
133
- minStep: 180,
134
- minValue: -90,
135
- maxValue: 90,
136
- validValues: [-90, 90],
137
- }).onGet(() => {
138
- return this.WindowCovering.TargetHorizontalTiltAngle
139
- }).onSet(this.TargetHorizontalTiltAngleSet.bind(this))
140
-
141
- // Initialize WindowCovering CurrentHorizontalTiltAngle Characteristic
142
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.CurrentHorizontalTiltAngle).setProps({
143
- minStep: 180,
144
- minValue: -90,
145
- maxValue: 90,
146
- validValues: [-90, 90],
147
- }).onGet(() => {
148
- return this.WindowCovering.CurrentHorizontalTiltAngle ?? 0
149
- })
150
-
151
- // Initialize Battery Service
152
- accessory.context.Battery = accessory.context.Battery ?? {}
153
- this.Battery = {
154
- Name: `${accessory.displayName} Battery`,
155
- Service: accessory.getService(this.hap.Service.Battery) ?? accessory.addService(this.hap.Service.Battery) as Service,
156
- BatteryLevel: accessory.context.BatteryLevel ?? 100,
157
- StatusLowBattery: this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_NORMAL,
158
- ChargingState: accessory.context.ChargingState ?? this.hap.Characteristic.ChargingState.NOT_CHARGING,
159
- }
160
- accessory.context.Battery = this.Battery as object
161
-
162
- // Initialize Battery Name Characteristic
163
- this.Battery.Service.setCharacteristic(this.hap.Characteristic.Name, this.Battery.Name).setCharacteristic(this.hap.Characteristic.ChargingState, this.hap.Characteristic.ChargingState.NOT_CHARGEABLE)
164
-
165
- // Initialize LightSensor Service
166
- if ((device as blindTiltConfig).hide_lightsensor) {
167
- if (this.LightSensor?.Service) {
168
- this.debugLog('Removing Light Sensor Service')
169
- this.LightSensor.Service = accessory.getService(this.hap.Service.LightSensor) as Service
170
- accessory.removeService(this.LightSensor.Service)
171
- accessory.context.LightSensor = {}
172
- } else {
173
- this.debugLog('Light Sensor Service is already removed')
174
- }
175
- } else {
176
- accessory.context.LightSensor = accessory.context.LightSensor ?? {}
177
- this.LightSensor = {
178
- Name: `${accessory.displayName} Light Sensor`,
179
- Service: accessory.getService(this.hap.Service.LightSensor) ?? accessory.addService(this.hap.Service.LightSensor) as Service,
180
- CurrentAmbientLightLevel: accessory.context.CurrentAmbientLightLevel ?? 0.0001,
181
- }
182
- accessory.context.LightSensor = this.LightSensor as object
183
-
184
- // Initialize LightSensor Characteristics
185
- this.LightSensor.Service.setCharacteristic(this.hap.Characteristic.Name, this.LightSensor.Name).setCharacteristic(this.hap.Characteristic.StatusActive, true).getCharacteristic(this.hap.Characteristic.CurrentAmbientLightLevel).onGet(() => {
186
- return this.LightSensor?.CurrentAmbientLightLevel ?? 0.0001
187
- })
188
- }
189
-
190
- // Initialize Open Mode Switch Service
191
- if (!(device as blindTiltConfig).silentModeSwitch) {
192
- if (this.OpenModeSwitch?.Service) {
193
- this.debugLog('Removing Open Mode Switch Service')
194
- this.OpenModeSwitch.Service = this.accessory.getService(this.hap.Service.Switch) as Service
195
- accessory.removeService(this.OpenModeSwitch.Service)
196
- accessory.context.OpenModeSwitch = {}
197
- }
198
- } else {
199
- accessory.context.OpenModeSwitch = accessory.context.OpenModeSwitch ?? {}
200
- this.OpenModeSwitch = {
201
- Name: `${accessory.displayName} Silent Open Mode`,
202
- Service: accessory.getService(this.hap.Service.Switch) ?? accessory.addService(this.hap.Service.Switch) as Service,
203
- On: accessory.context.OpenModeSwitch.On ?? false,
204
- }
205
- accessory.context.OpenModeSwitch = this.OpenModeSwitch as object
206
-
207
- // Initialize Open Mode Switch Service
208
- this.OpenModeSwitch.Service.setCharacteristic(this.hap.Characteristic.Name, this.OpenModeSwitch.Name).getCharacteristic(this.hap.Characteristic.On).onGet(() => {
209
- return this.OpenModeSwitch?.On ?? false
210
- })
211
-
212
- this.OpenModeSwitch.Service.getCharacteristic(this.hap.Characteristic.On).onSet(this.OpenModeSwitchSet.bind(this))
213
- }
214
-
215
- // Initialize Close Mode Switch Service
216
- if (!(device as blindTiltConfig).silentModeSwitch) {
217
- if (this.CloseModeSwitch?.Service) {
218
- this.debugLog('Removing Close Mode Switch Service')
219
- this.CloseModeSwitch.Service = this.accessory.getService(this.hap.Service.Switch) as Service
220
- accessory.removeService(this.CloseModeSwitch.Service)
221
- accessory.context.CloseModeSwitch = {}
222
- }
223
- } else {
224
- accessory.context.CloseModeSwitch = accessory.context.CloseModeSwitch ?? {}
225
- this.CloseModeSwitch = {
226
- Name: `${accessory.displayName} Silent Close Mode`,
227
- Service: accessory.getService(this.hap.Service.Switch) ?? accessory.addService(this.hap.Service.Switch) as Service,
228
- On: accessory.context.CloseModeSwitch.On ?? false,
229
- }
230
- accessory.context.CloseModeSwitch = this.CloseModeSwitch as object
231
-
232
- // Initialize Close Mode Switch Service
233
- this.CloseModeSwitch.Service.setCharacteristic(this.hap.Characteristic.Name, this.CloseModeSwitch.Name).getCharacteristic(this.hap.Characteristic.On).onGet(() => {
234
- return this.CloseModeSwitch?.On ?? false
235
- })
236
-
237
- this.CloseModeSwitch.Service.getCharacteristic(this.hap.Characteristic.On).onSet(this.CloseModeSwitchSet.bind(this))
238
- }
239
-
240
- // Retrieve initial values and updateHomekit
241
- try {
242
- this.debugLog('Retrieve initial values and update Homekit')
243
- this.refreshStatus()
244
- } catch (e: any) {
245
- this.errorLog(`failed to retrieve initial values and update Homekit, Error: ${e.message ?? e}`)
246
- }
247
-
248
- // regisiter webhook event handler if enabled
249
- try {
250
- this.debugLog('Registering Webhook Event Handler')
251
- this.registerWebhook()
252
- } catch (e: any) {
253
- this.errorLog(`failed to registerWebhook, Error: ${e.message ?? e}`)
254
- }
255
-
256
- // regisiter platform BLE event handler if enabled
257
- try {
258
- this.debugLog('Registering Platform BLE Event Handler')
259
- this.registerPlatformBLE()
260
- } catch (e: any) {
261
- this.errorLog(`failed to registerPlatformBLE, Error: ${e.message ?? e}`)
262
- }
263
-
264
- // Start an update interval
265
- interval(this.deviceRefreshRate * 1000)
266
- .pipe(skipWhile(() => this.blindTiltUpdateInProgress))
267
- .subscribe(async () => {
268
- await this.refreshStatus()
269
- })
270
-
271
- // update slide progress
272
- interval(this.deviceUpdateRate * 1000)
273
- .pipe(skipWhile(() => !this.blindTiltMoving))
274
- .subscribe(async () => {
275
- if (this.WindowCovering.PositionState === this.hap.Characteristic.PositionState.STOPPED) {
276
- return
277
- }
278
- this.debugLog(`Refresh Status When Moving, PositionState: ${this.WindowCovering.PositionState}`)
279
- await this.refreshStatus()
280
- })
281
-
282
- // Watch for BlindTilt change events
283
- // We put in a debounce of 100ms so we don't make duplicate calls
284
- this.doBlindTiltUpdate
285
- .pipe(
286
- tap(() => {
287
- this.blindTiltUpdateInProgress = true
288
- }),
289
- debounceTime(this.devicePushRate * 1000),
290
- )
291
- .subscribe(async () => {
292
- try {
293
- await this.pushChanges()
294
- } catch (e: any) {
295
- await this.apiError(e)
296
- this.errorLog(`failed pushChanges with ${device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`)
297
- }
298
- this.blindTiltUpdateInProgress = false
299
- })
300
- }
301
-
302
- /**
303
- * Parse the device status from the SwitchBotBLE API
304
- */
305
- async BLEparseStatus(): Promise<void> {
306
- this.debugLog('BLEparseStatus')
307
- this.debugLog(`(direction, slidePosition, battery, version) = BLE:(${this.serviceData.tilt}, ${this.serviceData.tilt}, ${this.serviceData.battery}, ${this.accessory.context.version}), current:(${this.WindowCovering.CurrentHorizontalTiltAngle}, ${this.WindowCovering.CurrentPosition}, ${this.Battery.BatteryLevel}, ${this.accessory.context.version})`)
308
-
309
- // CurrentPosition
310
- this.WindowCovering.CurrentPosition = 100 - Number(this.serviceData.tilt)
311
- await this.setMinMax()
312
- this.debugLog(`CurrentPosition ${this.WindowCovering.CurrentPosition}`)
313
- if (this.setNewTarget) {
314
- this.infoLog('Checking Status ...')
315
- }
316
- if (this.setNewTarget && this.serviceData.inMotion) {
317
- this.blindTiltMoving = true
318
- await this.setMinMax()
319
- if (Number(this.WindowCovering.TargetPosition) > this.WindowCovering.CurrentPosition) {
320
- this.debugLog(`Closing, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
321
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.INCREASING
322
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.PositionState).updateValue(this.WindowCovering.PositionState)
323
- this.debugLog(`Increasing, PositionState: ${this.WindowCovering.PositionState}`)
324
- } else if (Number(this.WindowCovering.TargetPosition) < this.WindowCovering.CurrentPosition) {
325
- this.debugLog(`Opening, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
326
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.DECREASING
327
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.PositionState).updateValue(this.WindowCovering.PositionState)
328
- this.debugLog(`Decreasing, PositionState: ${this.WindowCovering.PositionState}`)
329
- } else {
330
- this.debugLog(`Standby, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
331
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.STOPPED
332
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.PositionState).updateValue(this.WindowCovering.PositionState)
333
- this.debugLog('Stopped, PositionState', this.WindowCovering.PositionState)
334
- }
335
- } else {
336
- this.blindTiltMoving = false
337
- this.debugLog(`Standby, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
338
- this.WindowCovering.TargetPosition = this.WindowCovering.CurrentPosition
339
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.STOPPED
340
- this.debugLog(`Stopped, PositionState: ${this.WindowCovering.PositionState}`)
341
- }
342
- this.debugLog(`CurrentPosition: ${this.WindowCovering.CurrentPosition}, TargetPosition: ${this.WindowCovering.TargetPosition}, PositionState: ${this.WindowCovering.PositionState}`)
343
-
344
- // CurrentAmbientLightLevel
345
- if (!(this.device as blindTiltConfig).hide_lightsensor && this.LightSensor?.Service) {
346
- const set_minLux = (this.device as blindTiltConfig).set_minLux ?? 1
347
- const set_maxLux = (this.device as blindTiltConfig).set_maxLux ?? 6001
348
- const spaceBetweenLevels = 9
349
-
350
- this.getLightLevel(this.serviceData.lightLevel, set_minLux, set_maxLux, spaceBetweenLevels)
351
- this.debugLog(`LightLevel: ${this.serviceData.lightLevel}, CurrentAmbientLightLevel: ${this.LightSensor!.CurrentAmbientLightLevel}`)
352
- }
353
- // Battery Info
354
- if ('battery' in this.serviceData) {
355
- // BatteryLevel
356
- this.Battery.BatteryLevel = this.serviceData.battery
357
- this.debugLog(`BatteryLevel: ${this.Battery.BatteryLevel}`)
358
- // StatusLowBattery
359
- this.Battery.StatusLowBattery = this.Battery.BatteryLevel < 10
360
- ? this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_LOW
361
- : this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_NORMAL
362
- this.debugLog(`StatusLowBattery: ${this.Battery.StatusLowBattery}`)
363
- }
364
- };
365
-
366
- /**
367
- * Parse the device status from the SwitchBot OpenAPI
368
- */
369
- async openAPIparseStatus(): Promise<void> {
370
- this.debugLog('openAPIparseStatus')
371
- this.debugLog(`(direction, slidePosition, battery, version) = OpenAPI:(${this.deviceStatus.direction}, ${this.deviceStatus.slidePosition}, ${this.deviceStatus.battery}, ${this.deviceStatus.version}), current:(${this.WindowCovering.CurrentHorizontalTiltAngle}, ${this.WindowCovering.CurrentPosition}, ${this.Battery.BatteryLevel}, ${this.accessory.context.version})`)
372
-
373
- // CurrentPosition
374
- await this.getCurrentPosttionDirection(this.deviceStatus.direction, this.deviceStatus.slidePosition)
375
-
376
- if (!(this.device as blindTiltConfig).hide_lightsensor && this.LightSensor?.Service) {
377
- const set_minLux = (this.device as blindTiltConfig).set_minLux ?? 1
378
- const set_maxLux = (this.device as blindTiltConfig).set_maxLux ?? 6001
379
- const lightLevel = this.deviceStatus.lightLevel === 'bright' ? set_maxLux : set_minLux
380
- this.LightSensor.CurrentAmbientLightLevel = this.getLightLevel(lightLevel, set_minLux, set_maxLux, 2)
381
- this.debugLog(`LightLevel: ${this.deviceStatus.lightLevel}, CurrentAmbientLightLevel: ${this.LightSensor.CurrentAmbientLightLevel}`)
382
- }
383
-
384
- // BatteryLevel
385
- this.Battery.BatteryLevel = this.deviceStatus.battery
386
- this.debugLog(`BatteryLevel: ${this.Battery.BatteryLevel}`)
387
-
388
- // StatusLowBattery
389
- this.Battery.StatusLowBattery = this.Battery.BatteryLevel < 10
390
- ? this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_LOW
391
- : this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_NORMAL
392
- this.debugLog(`StatusLowBattery: ${this.Battery.StatusLowBattery}`)
393
-
394
- // Firmware Version
395
- const version = this.deviceStatus.version.toString()
396
- this.debugLog(`Firmware Version: ${version.replace(/^V|-.*$/g, '')}`)
397
- let deviceVersion: string
398
- if (version?.includes('.') === false) {
399
- const replace = version?.replace(/^V|-.*$/g, '')
400
- const match = replace?.match(/./g)
401
- const blindTiltVersion = match?.join('.') ?? '0.0.0'
402
- deviceVersion = blindTiltVersion
403
- } else {
404
- deviceVersion = version.replace(/^V|-.*$/g, '') ?? '0.0.0'
405
- }
406
- this.accessory
407
- .getService(this.hap.Service.AccessoryInformation)!
408
- .setCharacteristic(this.hap.Characteristic.HardwareRevision, deviceVersion)
409
- .setCharacteristic(this.hap.Characteristic.FirmwareRevision, deviceVersion)
410
- .getCharacteristic(this.hap.Characteristic.FirmwareRevision)
411
- .updateValue(deviceVersion)
412
- this.accessory.context.version = deviceVersion
413
- this.debugLog(`version: ${this.accessory.context.version}`)
414
- }
415
-
416
- async parseStatusWebhook(): Promise<void> {
417
- this.debugLog('parseStatusWebhook')
418
- this.debugLog(`(slidePosition, battery, version) = Webhook:(${this.webhookContext.direction}, ${this.webhookContext.slidePosition}, ${this.webhookContext.battery}, ${this.webhookContext.version}, current:(${this.WindowCovering.CurrentHorizontalTiltAngle}, ${this.WindowCovering.CurrentPosition}, ${this.Battery.BatteryLevel}, ${this.accessory.context.version})`)
419
- // CurrentPosition and CurrentHorizontalTiltAngle
420
- await this.getCurrentPosttionDirection(this.webhookContext.direction, this.webhookContext.slidePosition)
421
- // BatteryLevel
422
- this.Battery.BatteryLevel = this.webhookContext.battery
423
- this.debugLog(`BatteryLevel: ${this.Battery.BatteryLevel}`)
424
- // StatusLowBattery
425
- this.Battery.StatusLowBattery = this.Battery.BatteryLevel < 10
426
- ? this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_LOW
427
- : this.hap.Characteristic.StatusLowBattery.BATTERY_LEVEL_NORMAL
428
- this.debugLog(`StatusLowBattery: ${this.Battery.StatusLowBattery}`)
429
- // Firmware Version
430
- const deviceVersion = this.webhookContext.version.replace(/^V|-.*$/g, '') ?? '0.0.0'
431
- this.accessory
432
- .getService(this.hap.Service.AccessoryInformation)!
433
- .setCharacteristic(this.hap.Characteristic.HardwareRevision, deviceVersion)
434
- .setCharacteristic(this.hap.Characteristic.FirmwareRevision, deviceVersion)
435
- .getCharacteristic(this.hap.Characteristic.FirmwareRevision)
436
- .updateValue(deviceVersion)
437
- this.accessory.context.version = deviceVersion
438
- this.debugSuccessLog(`version: ${this.accessory.context.version}`)
439
- }
440
-
441
- /**
442
- * Asks the SwitchBot API for the latest device information
443
- */
444
- async refreshStatus(): Promise<void> {
445
- if (this.BLE) {
446
- await this.BLERefreshStatus()
447
- } else if (this.OpenAPI && this.platform.config.credentials?.token) {
448
- await this.openAPIRefreshStatus()
449
- } else {
450
- await this.offlineOff()
451
- this.debugWarnLog(`Connection Type: ${this.device.connectionType}, refreshStatus will not happen.`)
452
- }
453
- }
454
-
455
- async BLERefreshStatus(): Promise<void> {
456
- this.debugLog('BLERefreshStatus')
457
- const switchBotBLE = await this.switchbotBLE()
458
- if (switchBotBLE === undefined) {
459
- await this.BLERefreshConnection(switchBotBLE)
460
- } else {
461
- // Start to monitor advertisement packets
462
- (async () => {
463
- // Start to monitor advertisement packets
464
- const serviceData = await this.monitorAdvertisementPackets(switchBotBLE) as blindTiltServiceData
465
- // Update HomeKit
466
- if (serviceData.model === SwitchBotBLEModel.BlindTilt && serviceData.modelName === SwitchBotBLEModelName.BlindTilt) {
467
- this.serviceData = serviceData
468
- if (serviceData !== undefined && serviceData !== null) {
469
- await this.BLEparseStatus()
470
- await this.updateHomeKitCharacteristics()
471
- } else {
472
- this.errorLog(`serviceData is either undefined or null, serviceData: ${JSON.stringify(serviceData)}`)
473
- await this.BLERefreshConnection(switchBotBLE)
474
- }
475
- } else {
476
- this.errorLog(`failed to get serviceData, serviceData: ${JSON.stringify(serviceData)}`)
477
- await this.BLERefreshConnection(switchBotBLE)
478
- }
479
- })()
480
- }
481
- }
482
-
483
- async registerPlatformBLE(): Promise<void> {
484
- this.debugLog('registerPlatformBLE')
485
- if (this.config.options?.BLE && !this.device.disablePlatformBLE) {
486
- this.debugLog('is listening to Platform BLE.')
487
- try {
488
- const formattedDeviceId = formatDeviceIdAsMac(this.device.deviceId)
489
- this.device.bleMac = formattedDeviceId
490
- this.debugLog(`bleMac: ${this.device.bleMac}`)
491
- this.platform.bleEventHandler[this.device.bleMac] = async (context: blindTiltServiceData) => {
492
- try {
493
- this.serviceData = context
494
- if (context !== undefined && context !== null) {
495
- this.debugLog(`received BLE: ${JSON.stringify(context)}`)
496
- await this.BLEparseStatus()
497
- await this.updateHomeKitCharacteristics()
498
- } else {
499
- this.errorLog(`context is either undefined or null, context: ${JSON.stringify(context)}`)
500
- await this.BLERefreshConnection(context)
501
- }
502
- } catch (e: any) {
503
- this.errorLog(`failed to handle BLE. Received: ${JSON.stringify(context)} Error: ${e.message ?? e}`)
504
- }
505
- }
506
- } catch (error) {
507
- this.errorLog(`failed to format device ID as MAC, Error: ${error}`)
508
- }
509
- } else {
510
- this.debugLog('is not listening to Platform BLE')
511
- }
512
- }
513
-
514
- async openAPIRefreshStatus(): Promise<void> {
515
- this.debugLog('openAPIRefreshStatus')
516
- try {
517
- const deviceStatus = await this.deviceRefreshStatus<blindTiltStatus>()
518
- this.debugLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
519
- if (await this.successfulStatusCodes(deviceStatus)) {
520
- this.debugSuccessLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
521
- this.deviceStatus = deviceStatus.body
522
- await this.openAPIparseStatus()
523
- await this.updateHomeKitCharacteristics()
524
- } else {
525
- this.debugWarnLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
526
- }
527
- } catch (e: any) {
528
- await this.apiError(e)
529
- this.errorLog(`failed openAPIRefreshStatus with ${this.device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`)
530
- }
531
- }
532
-
533
- async registerWebhook(): Promise<void> {
534
- if (this.device.webhook) {
535
- this.debugLog('is listening webhook.')
536
- this.platform.webhookEventHandler[this.device.deviceId] = async (context: blindTiltWebhookContext) => {
537
- try {
538
- this.webhookContext = context
539
- if (context !== undefined && context !== null) {
540
- this.debugLog(`received Webhook: ${JSON.stringify(context)}`)
541
- await this.parseStatusWebhook()
542
- await this.updateHomeKitCharacteristics()
543
- } else {
544
- this.errorLog(`context is either undefined or null, context: ${JSON.stringify(context)}`)
545
- }
546
- } catch (e: any) {
547
- this.errorLog(`failed to handle webhook. Received: ${JSON.stringify(context)} Error: ${e.message ?? e}`)
548
- }
549
- }
550
- } else {
551
- this.debugLog('is not listening webhook.')
552
- }
553
- }
554
-
555
- async pushChanges(): Promise<void> {
556
- if (this.BLE) {
557
- await this.BLEpushChanges()
558
- } else if (this.OpenAPI && this.platform.config.credentials?.token) {
559
- await this.openAPIpushChanges()
560
- } else {
561
- await this.offlineOff()
562
- this.debugWarnLog(`Connection Type: ${this.device.connectionType}, pushChanges will not happen.`)
563
- }
564
- // Refresh the status from the API
565
- interval(15000)
566
- .pipe(skipWhile(() => this.blindTiltUpdateInProgress))
567
- .pipe(take(1))
568
- .subscribe(async () => {
569
- await this.refreshStatus()
570
- })
571
- }
572
-
573
- async BLEpushChanges(): Promise<void> {
574
- this.debugLog('BLEpushChanges')
575
- if (this.WindowCovering.TargetPosition !== this.WindowCovering.CurrentPosition) {
576
- this.debugLog(`BLEpushChanges On: ${this.WindowCovering.TargetPosition} OnCached: ${this.WindowCovering.CurrentPosition}`)
577
- const switchBotBLE = await this.platform.connectBLE(this.accessory, this.device)
578
- try {
579
- const formattedDeviceId = formatDeviceIdAsMac(this.device.deviceId)
580
- this.device.bleMac = formattedDeviceId
581
- this.debugLog(`bleMac: ${this.device.bleMac}`)
582
- const { setPositionMode, Mode }: { setPositionMode: number, Mode: string } = await this.setPerformance()
583
- this.debugLog(`Mode: ${Mode}, setPositionMode: ${setPositionMode}`)
584
- if (switchBotBLE !== false) {
585
- switchBotBLE
586
- .discover({ model: this.device.bleModel, quick: true, id: this.device.bleMac })
587
- .then(async (device_list: SwitchbotDevice[]) => {
588
- const deviceList = device_list as WoBlindTilt[]
589
- return await this.retryBLE({
590
- max: this.maxRetryBLE(),
591
- fn: async () => {
592
- if (deviceList && Array.isArray(deviceList) && deviceList.length > 0) {
593
- return await deviceList[0].runToPos(100 - Number(this.WindowCovering.TargetPosition), setPositionMode)
594
- } else {
595
- throw new Error('No device found')
596
- }
597
- },
598
- })
599
- })
600
- .then(async () => {
601
- this.successLog(`TargetPostion: ${this.WindowCovering.TargetPosition} sent over SwitchBot BLE, sent successfully`)
602
- await this.updateHomeKitCharacteristics()
603
- })
604
- .catch(async (e: any) => {
605
- await this.apiError(e)
606
- this.errorLog(`failed BLEpushChanges with ${this.device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`)
607
- await this.BLEPushConnection()
608
- })
609
- } else {
610
- this.errorLog(`wasn't able to establish BLE Connection, node-switchbot: ${JSON.stringify(switchBotBLE)}`)
611
- await this.BLEPushConnection()
612
- }
613
- } catch (error) {
614
- this.errorLog(`failed to format device ID as MAC, Error: ${error}`)
615
- }
616
- } else {
617
- this.debugLog(`No changes (BLEpushChanges), TargetPosition: ${this.WindowCovering.TargetPosition}, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
618
- }
619
- }
620
-
621
- async openAPIpushChanges(): Promise<void> {
622
- this.debugLog('openAPIpushChanges')
623
- const hasDifferentAndRelevantHorizontalTiltAngle
624
- = this.mappingMode === BlindTiltMappingMode.UseTiltForDirection
625
- && this.WindowCovering.TargetHorizontalTiltAngle !== this.WindowCovering.CurrentHorizontalTiltAngle
626
- if (this.WindowCovering.TargetPosition !== this.WindowCovering.CurrentPosition
627
- || hasDifferentAndRelevantHorizontalTiltAngle || this.device.disableCaching) {
628
- const [direction, position] = this.mapHomekitValuesToDeviceValues(Number(this.WindowCovering.TargetPosition), Number(this.WindowCovering.TargetHorizontalTiltAngle))
629
- const { Mode, setPositionMode }: { setPositionMode: number, Mode: string } = await this.setPerformance()
630
- this.debugLog(`Pushing ${this.WindowCovering.TargetPosition} (device = ${direction};${position})`)
631
- this.debugLog(`Mode: ${Mode}, setPositionMode: ${setPositionMode}`)
632
- let bodyChange: bodyChange
633
- if (position === 100) {
634
- bodyChange = {
635
- command: 'fullyOpen',
636
- parameter: 'default',
637
- commandType: 'command',
638
- }
639
- } else if (position === 0) {
640
- bodyChange = {
641
- command: direction === 'up' ? 'closeUp' : 'closeDown',
642
- parameter: 'default',
643
- commandType: 'command',
644
- }
645
- } else {
646
- bodyChange = {
647
- command: 'setPosition',
648
- parameter: `${direction};${position}`,
649
- commandType: 'command',
650
- }
651
- }
652
- this.debugLog(`SwitchBot OpenAPI bodyChange: ${JSON.stringify(bodyChange)}`)
653
- try {
654
- const deviceStatus = await this.pushChangeRequest(bodyChange)
655
- this.debugLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
656
- if (await this.successfulStatusCodes(deviceStatus)) {
657
- this.debugSuccessLog(`statusCode: ${deviceStatus.statusCode}, deviceStatus: ${JSON.stringify(deviceStatus)}`)
658
- await this.updateHomeKitCharacteristics()
659
- } else {
660
- await this.statusCode(deviceStatus.statusCode)
661
- }
662
- } catch (e: any) {
663
- await this.apiError(e)
664
- this.errorLog(`failed openAPIpushChanges with ${this.device.connectionType} Connection, Error Message: ${JSON.stringify(e.message)}`)
665
- }
666
- } else {
667
- this.debugLog(`No changes (openAPIpushChanges), TargetPosition: ${this.WindowCovering.TargetPosition}, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
668
- this.debugLog(`No changes (openAPIpushChanges), TargetHorizontalTiltAngle: ${this.WindowCovering.TargetHorizontalTiltAngle}, CurrentHorizontalTiltAngle: ${this.WindowCovering.CurrentHorizontalTiltAngle}`)
669
- }
670
- }
671
-
672
- /**
673
- * Handle requests to set the value of the "Target Horizontal Tilt" characteristic
674
- */
675
- async TargetHorizontalTiltAngleSet(value: CharacteristicValue): Promise<void> {
676
- if (this.WindowCovering.TargetHorizontalTiltAngle !== this.accessory.context.TargetHorizontalTiltAngle) {
677
- this.debugLog(`Set TargetHorizontalTiltAngle: ${value}`)
678
- } else {
679
- this.debugLog(`No changes, TargetHorizontalTiltAngle: ${value}`)
680
- }
681
-
682
- // value = value < 0 ? -90 : 90;
683
- this.WindowCovering.TargetHorizontalTiltAngle = value
684
- await this.mqtt('TargetHorizontalTiltAngle', this.WindowCovering.TargetHorizontalTiltAngle)
685
- await this.startUpdatingBlindTiltIfNeeded()
686
- }
687
-
688
- /**
689
- * Handle requests to set the value of the "Target Position" characteristic
690
- */
691
- async TargetPositionSet(value: CharacteristicValue): Promise<void> {
692
- if (this.WindowCovering.TargetPosition !== this.accessory.context.TargetPosition) {
693
- this.debugLog(`Set TargetPosition: ${value}`)
694
- } else {
695
- this.debugLog(`No changes, TargetPosition: ${value}`)
696
- }
697
-
698
- this.WindowCovering.TargetPosition = value
699
- await this.mqtt('TargetPosition', this.WindowCovering.TargetPosition)
700
- await this.startUpdatingBlindTiltIfNeeded()
701
- }
702
-
703
- async startUpdatingBlindTiltIfNeeded(): Promise<void> {
704
- await this.setMinMax()
705
- this.debugLog('setMinMax')
706
- if (this.WindowCovering.TargetPosition > this.WindowCovering.CurrentPosition
707
- || this.WindowCovering.TargetHorizontalTiltAngle !== this.WindowCovering.CurrentHorizontalTiltAngle) {
708
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.INCREASING
709
- this.setNewTarget = true
710
- this.debugLog(`value: ${this.WindowCovering.CurrentPosition}, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
711
- } else if (this.WindowCovering.TargetPosition < this.WindowCovering.CurrentPosition) {
712
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.DECREASING
713
- this.setNewTarget = true
714
- this.debugLog(`value: ${this.WindowCovering.CurrentPosition}, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
715
- } else {
716
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.STOPPED
717
- this.setNewTarget = false
718
- this.debugLog(`value: ${this.WindowCovering.CurrentPosition}, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
719
- }
720
- this.WindowCovering.Service.setCharacteristic(this.hap.Characteristic.PositionState, this.WindowCovering.PositionState)
721
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.PositionState).updateValue(this.WindowCovering.PositionState)
722
-
723
- /**
724
- * If Blind Tilt movement time is short, the moving flag from backend is always false.
725
- * The minimum time depends on the network control latency.
726
- */
727
- clearTimeout(this.setNewTargetTimer)
728
- this.debugLog(`deviceUpdateRate: ${this.deviceUpdateRate}`)
729
- if (this.setNewTarget) {
730
- this.setNewTargetTimer = setTimeout(async () => {
731
- this.debugLog(`setNewTarget ${this.setNewTarget} timeout`)
732
- this.setNewTarget = false
733
- }, this.deviceUpdateRate * 1000)
734
- }
735
- this.doBlindTiltUpdate.next()
736
- }
737
-
738
- /**
739
- * Handle requests to set the value of the "Target Position" characteristic
740
- */
741
- async OpenModeSwitchSet(value: CharacteristicValue): Promise<void> {
742
- if (this.OpenModeSwitch && (this.device as blindTiltConfig).silentModeSwitch) {
743
- this.debugLog(`Silent Open Mode: ${value}`)
744
- this.OpenModeSwitch.On = value
745
- this.accessory.context.OpenModeSwitch.On = value
746
- this.doBlindTiltUpdate.next()
747
- }
748
- }
749
-
750
- /**
751
- * Handle requests to set the value of the "Target Position" characteristic
752
- */
753
- async CloseModeSwitchSet(value: CharacteristicValue): Promise<void> {
754
- if (this.CloseModeSwitch && (this.device as blindTiltConfig).silentModeSwitch) {
755
- this.debugLog(`Silent Close Mode: ${value}`)
756
- this.CloseModeSwitch.On = value
757
- this.accessory.context.CloseModeSwitch.On = value
758
- this.doBlindTiltUpdate.next()
759
- }
760
- }
761
-
762
- async updateHomeKitCharacteristics(): Promise<void> {
763
- await this.setMinMax()
764
- // CurrentHorizontalTiltAngle
765
- if (this.mappingMode === BlindTiltMappingMode.UseTiltForDirection) {
766
- await this.updateCharacteristic(this.WindowCovering.Service, this.hap.Characteristic.CurrentHorizontalTiltAngle, this.WindowCovering.CurrentHorizontalTiltAngle, 'CurrentHorizontalTiltAngle')
767
- }
768
- // CurrentPosition
769
- await this.updateCharacteristic(this.WindowCovering.Service, this.hap.Characteristic.CurrentPosition, this.WindowCovering.CurrentPosition, 'CurrentPosition')
770
- // PositionState
771
- await this.updateCharacteristic(this.WindowCovering.Service, this.hap.Characteristic.PositionState, this.WindowCovering.PositionState, 'PositionState')
772
- // TargetPosition
773
- await this.updateCharacteristic(this.WindowCovering.Service, this.hap.Characteristic.TargetPosition, this.WindowCovering.TargetPosition, 'TargetPosition')
774
- // CurrentAmbientLightLevel
775
- if (!(this.device as blindTiltConfig).hide_lightsensor && this.LightSensor?.Service) {
776
- const history = { time: Math.round(new Date().valueOf() / 1000), lux: this.LightSensor.CurrentAmbientLightLevel }
777
- await this.updateCharacteristic(this.LightSensor?.Service, this.hap.Characteristic.CurrentAmbientLightLevel, this.LightSensor?.CurrentAmbientLightLevel, 'CurrentAmbientLightLevel', history)
778
- }
779
- // BatteryLevel
780
- await this.updateCharacteristic(this.Battery.Service, this.hap.Characteristic.BatteryLevel, this.Battery.BatteryLevel, 'BatteryLevel')
781
- // StatusLowBattery
782
- await this.updateCharacteristic(this.Battery.Service, this.hap.Characteristic.StatusLowBattery, this.Battery.StatusLowBattery, 'StatusLowBattery')
783
- // ChargingState
784
- await this.updateCharacteristic(this.Battery.Service, this.hap.Characteristic.ChargingState, this.Battery.ChargingState, 'ChargingState')
785
- }
786
-
787
- async BLEPushConnection() {
788
- if (this.platform.config.credentials?.token && this.device.connectionType === 'BLE/OpenAPI') {
789
- this.warnLog('Using OpenAPI Connection to Push Changes')
790
- await this.openAPIpushChanges()
791
- }
792
- }
793
-
794
- async BLERefreshConnection(switchbot: SwitchBotBLE): Promise<void> {
795
- this.errorLog(`wasn't able to establish BLE Connection, node-switchbot: ${switchbot}`)
796
- if (this.platform.config.credentials?.token && this.device.connectionType === 'BLE/OpenAPI') {
797
- this.warnLog('Using OpenAPI Connection to Refresh Status')
798
- await this.openAPIRefreshStatus()
799
- }
800
- }
801
-
802
- async setPerformance() {
803
- let setPositionMode: number
804
- let Mode: string
805
- if (Number(this.WindowCovering.TargetPosition) > 50) {
806
- if ((this.device as blindTiltConfig).setOpenMode === '1' || this.OpenModeSwitch?.On) {
807
- setPositionMode = 1
808
- Mode = 'Silent Mode'
809
- } else if ((this.device as blindTiltConfig).setOpenMode === '0' || !this.OpenModeSwitch?.On) {
810
- setPositionMode = 0
811
- Mode = 'Performance Mode'
812
- } else {
813
- setPositionMode = 0
814
- Mode = 'Default Mode'
815
- }
816
- } else {
817
- if ((this.device as blindTiltConfig).setCloseMode === '1' || this.CloseModeSwitch?.On) {
818
- setPositionMode = 1
819
- Mode = 'Silent Mode'
820
- } else if ((this.device as blindTiltConfig).setOpenMode === '0' || !this.CloseModeSwitch?.On) {
821
- setPositionMode = 0
822
- Mode = 'Performance Mode'
823
- } else {
824
- setPositionMode = 0
825
- Mode = 'Default Mode'
826
- }
827
- }
828
- return { setPositionMode, Mode }
829
- }
830
-
831
- async setMinMax(): Promise<void> {
832
- if ((this.device as blindTiltConfig).set_min) {
833
- if (Number(this.WindowCovering.CurrentPosition) <= (this.device as blindTiltConfig).set_min!) {
834
- this.WindowCovering.CurrentPosition = 0
835
- }
836
- }
837
- if ((this.device as blindTiltConfig).set_max) {
838
- if (Number(this.WindowCovering.CurrentPosition) >= (this.device as blindTiltConfig).set_max!) {
839
- this.WindowCovering.CurrentPosition = 100
840
- }
841
- }
842
- if (this.device.history) {
843
- const motion = this.accessory.getService(this.hap.Service.MotionSensor)
844
- const state = Number(this.WindowCovering.CurrentPosition) > 0 ? 1 : 0
845
- motion?.updateCharacteristic(this.hap.Characteristic.MotionDetected, state)
846
- }
847
-
848
- if (this.mappingMode === BlindTiltMappingMode.UseTiltForDirection) {
849
- this.WindowCovering.CurrentHorizontalTiltAngle = Number(this.WindowCovering.CurrentHorizontalTiltAngle) < 0 ? -90 : 90
850
- }
851
- }
852
-
853
- async offlineOff(): Promise<void> {
854
- if (this.device.offline) {
855
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.CurrentPosition, 100)
856
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.PositionState, this.hap.Characteristic.PositionState.STOPPED)
857
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.TargetPosition, 100)
858
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.CurrentHorizontalTiltAngle, 90)
859
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.TargetHorizontalTiltAngle, 90)
860
- }
861
- }
862
-
863
- async apiError(e: any): Promise<void> {
864
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.CurrentPosition, e)
865
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.PositionState, e)
866
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.TargetPosition, e)
867
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.CurrentHorizontalTiltAngle, e)
868
- this.WindowCovering.Service.updateCharacteristic(this.hap.Characteristic.TargetHorizontalTiltAngle, e)
869
- this.Battery.Service.updateCharacteristic(this.hap.Characteristic.BatteryLevel, e)
870
- this.Battery.Service.updateCharacteristic(this.hap.Characteristic.StatusLowBattery, e)
871
- this.Battery.Service.updateCharacteristic(this.hap.Characteristic.ChargingState, e)
872
- if (!(this.device as blindTiltConfig).hide_lightsensor && this.LightSensor?.Service) {
873
- this.LightSensor.Service.updateCharacteristic(this.hap.Characteristic.CurrentAmbientLightLevel, e)
874
- this.LightSensor.Service.updateCharacteristic(this.hap.Characteristic.StatusActive, e)
875
- }
876
- }
877
-
878
- async getCurrentPosttionDirection(
879
- direction: blindTiltStatus['direction'] | blindTiltWebhookContext['direction'],
880
- slidePosition: blindTiltStatus['slidePosition'] | blindTiltWebhookContext['slidePosition'],
881
- ) {
882
- const [homekitPosition, homekitTiltAngle] = this.mapDeviceValuesToHomekitValues(Number(slidePosition), String(direction))
883
- this.debugLog(`Slide Position: ${slidePosition}`)
884
- this.debugLog(`Homekit Position: ${homekitPosition}`)
885
-
886
- this.WindowCovering.CurrentPosition = homekitPosition
887
- await this.setMinMax()
888
- this.debugLog(`CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
889
-
890
- if (homekitTiltAngle) {
891
- this.WindowCovering.CurrentHorizontalTiltAngle = homekitTiltAngle!
892
- this.debugLog(`CurrentHorizontalTiltAngle: ${this.WindowCovering.CurrentHorizontalTiltAngle}`)
893
- }
894
-
895
- if (this.setNewTarget) {
896
- this.blindTiltMoving = true
897
- this.infoLog('Checking Status ...')
898
- await this.setMinMax()
899
- if (this.WindowCovering.TargetPosition > this.WindowCovering.CurrentPosition
900
- || (homekitTiltAngle && this.WindowCovering.TargetHorizontalTiltAngle !== this.WindowCovering.CurrentHorizontalTiltAngle)) {
901
- this.debugLog(`Closing, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
902
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.INCREASING
903
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.PositionState).updateValue(this.WindowCovering.PositionState)
904
- this.debugLog(`Increasing, PositionState: ${this.WindowCovering.PositionState}`)
905
- } else if (this.WindowCovering.TargetPosition < this.WindowCovering.CurrentPosition) {
906
- this.debugLog(`Opening, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
907
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.DECREASING
908
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.PositionState).updateValue(this.WindowCovering.PositionState)
909
- this.debugLog(`Decreasing, PositionState: ${this.WindowCovering.PositionState}`)
910
- } else {
911
- this.debugLog(`Standby because reached position, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
912
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.STOPPED
913
- this.WindowCovering.Service.getCharacteristic(this.hap.Characteristic.PositionState).updateValue(this.WindowCovering.PositionState)
914
- this.debugLog(`Stopped, PositionState: ${this.WindowCovering.PositionState}`)
915
- }
916
- } else {
917
- this.blindTiltMoving = false
918
- this.debugLog(`Standby because device not moving, CurrentPosition: ${this.WindowCovering.CurrentPosition}`)
919
- this.WindowCovering.TargetPosition = this.WindowCovering.CurrentPosition
920
- if (homekitTiltAngle) {
921
- this.WindowCovering.TargetHorizontalTiltAngle = this.WindowCovering.CurrentHorizontalTiltAngle
922
- }
923
- this.WindowCovering.PositionState = this.hap.Characteristic.PositionState.STOPPED
924
- this.debugLog(`Stopped, PositionState: ${this.WindowCovering.PositionState}`)
925
- }
926
- this.debugLog(`CurrentPosition: ${this.WindowCovering.CurrentPosition}, TargetPosition: ${this.WindowCovering.TargetPosition}, PositionState: ${this.WindowCovering.PositionState}`)
927
- }
928
-
929
- /**
930
- * Maps device values to homekit values
931
- *
932
- * @param devicePosition the position as reported by the devide
933
- * @param deviceDirection the direction as reported by the device
934
- * @returns [homekit position, homekit tiltAngle]
935
- */
936
- mapDeviceValuesToHomekitValues(devicePosition: number, deviceDirection: string): [CharacteristicValue, CharacteristicValue?] {
937
- // device position 0 => closed down
938
- // device position 50 => open
939
- // device position 100 => closed up
940
-
941
- // homekit position 0 => closed
942
- // homekit position 100 => open
943
- const direction = deviceDirection === 'up' ? 'up' : 'down'
944
- this.debugLog(`Mapping device values to homekit values, devicePostion: ${devicePosition}, deviceDirection: ${direction}`)
945
- switch (this.mappingMode) {
946
- case BlindTiltMappingMode.OnlyUp:
947
- // we only close upwards, so we see anything that is tilted downwards(<50) as open
948
- if (devicePosition < 50) {
949
- return [100, undefined] // fully open in homekit
950
- } else {
951
- // we range from 50->100, with 100 being closed, so map to homekit by scaling to 0..100 and then reversing
952
- return [100 - (devicePosition - 50) * 2, undefined]
953
- }
954
-
955
- case BlindTiltMappingMode.OnlyDown:
956
- // we only close downwards, so we see anything that is tilted upwards(>50) as upwards
957
- if (devicePosition > 50) {
958
- return [100, undefined] // fully open in homekit
959
- } else {
960
- // we range from 0..50 so scale to homekit and then reverse
961
- return [devicePosition * 2, undefined]
962
- }
963
-
964
- case BlindTiltMappingMode.DownAndUp:
965
- // we close both ways with closed downwards being 0 in homekit and closed upwards in homekit being 100. Open is 50 in homekit
966
- return [devicePosition, undefined]
967
-
968
- case BlindTiltMappingMode.UpAndDown:
969
- // we close both ways with closed downwards being 1000 in homekit and closed upwards in homekit being 0. Open is 50 in homekit.,
970
- // so we reverse the value
971
- return [100 - devicePosition, undefined]
972
-
973
- case BlindTiltMappingMode.UseTiltForDirection:
974
- // we use tilt for direction, so being closed downwards is 0 in homekit with -90 tilt, while being closed upwards is 0 with 90 tilt.
975
- if (devicePosition <= 50) {
976
- // downwards tilted, so we range from 0..50, with 0 being closed and 50 being open, so scale.
977
- return [devicePosition * 2, -90]
978
- } else {
979
- // upwards tilted, so we range from 50..100, with 50 being open and 100 being closed, so scale and rever
980
- return [100 - (devicePosition - 50) * 2, 90]
981
- }
982
- }
983
- }
984
-
985
- /**
986
- * Maps homekit values to device values
987
- *
988
- * @param homekitPosition the position as reported by homekit
989
- * @param homekitTiltAngle the tilt angle as reported by homekit
990
- * @returns [device position, device direction]
991
- */
992
- mapHomekitValuesToDeviceValues(homekitPosition: number, homekitTiltAngle: number): [string, number] {
993
- // homekit position 0 => closed
994
- // homekit position 100 => open
995
-
996
- // device position [up, 0] = closed upwards
997
- // device position [down, 0] = closed downwards
998
- // device position [up, 100] = open
999
- // device position [down, 100] = open
1000
-
1001
- switch (this.mappingMode) {
1002
- case BlindTiltMappingMode.OnlyUp:
1003
- // invert
1004
- return ['up', homekitPosition]
1005
- case BlindTiltMappingMode.OnlyDown:
1006
- // invert
1007
- return ['down', homekitPosition]
1008
-
1009
- case BlindTiltMappingMode.DownAndUp:
1010
- // homekit 0 = downwards closed,
1011
- // homekit 50 = open,
1012
- // homekit 100 = upwards closed
1013
- if (homekitPosition <= 50) {
1014
- // homekit 0..50 -> device 100..0 so scale and invert
1015
- return ['down', 100 - homekitPosition * 2]
1016
- } else {
1017
- // homekit 50..100 -> device 0..100, so rebase, scale and invert
1018
- return ['up', (homekitPosition - 50) * 2]
1019
- }
1020
-
1021
- case BlindTiltMappingMode.UpAndDown:
1022
- // homekit 0 = upwards closed,
1023
- // homekit 50 = open,
1024
- // homekit 100 = upwards closed
1025
- if (homekitPosition <= 50) {
1026
- // homekit 0..50 -> device 0..100 so scale and invert
1027
- return ['up', homekitPosition * 2]
1028
- } else {
1029
- // homekit 50..100 -> device 100...0 so scale
1030
- return ['down', 100 - homekitPosition * 2]
1031
- }
1032
-
1033
- case BlindTiltMappingMode.UseTiltForDirection:
1034
- // tilt -90, homekit 0 = closed downwards
1035
- // tilt -90, homekit 100 = open
1036
- // tilt 90, homekit 0 = closed upwards
1037
- // tilt 90, homekit 100 = open
1038
- if (homekitTiltAngle! <= 0) {
1039
- // downwards
1040
- // homekit 0..100 -> device 0..100, so invert
1041
- return ['down', homekitPosition]
1042
- } else {
1043
- // upwards
1044
- // homekit 0..100 -> device 0..100, so invert
1045
- return ['up', homekitPosition]
1046
- }
1047
- }
1048
- }
1049
- }