@switchbot/homebridge-switchbot 5.0.0-beta.9 → 5.0.0-beta.90

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