@switchbot/homebridge-switchbot 5.0.0-beta.7 → 5.0.0-beta.71
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.github/ISSUE_TEMPLATE/e2e-verification.md +36 -0
- package/.github/workflows/ci.yml +61 -0
- package/.github/workflows/manual-e2e.yml +103 -0
- package/CHANGELOG.md +35 -0
- package/E2E-VERIFICATION.md +121 -0
- package/MIGRATION.md +44 -0
- package/README.md +56 -3
- package/config.schema.json +99 -14823
- package/dist/deviceFactory.d.ts +13 -0
- package/dist/deviceFactory.d.ts.map +1 -0
- package/dist/deviceFactory.js +81 -0
- package/dist/deviceFactory.js.map +1 -0
- package/dist/devices/deviceBase.d.ts +50 -0
- package/dist/devices/deviceBase.d.ts.map +1 -0
- package/dist/devices/deviceBase.js +119 -0
- package/dist/devices/deviceBase.js.map +1 -0
- package/dist/devices/genericDevice.d.ts +283 -0
- package/dist/devices/genericDevice.d.ts.map +1 -0
- package/dist/devices/genericDevice.js +1035 -0
- package/dist/devices/genericDevice.js.map +1 -0
- package/dist/homebridge-ui/public/index.html +77 -245
- package/dist/homebridge-ui/server.d.ts +3 -1
- package/dist/homebridge-ui/server.d.ts.map +1 -1
- package/dist/homebridge-ui/server.js +42 -36
- package/dist/homebridge-ui/server.js.map +1 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +5 -12
- package/dist/index.js.map +1 -1
- package/dist/platform.d.ts +27 -0
- package/dist/platform.d.ts.map +1 -0
- package/dist/platform.js +404 -0
- package/dist/platform.js.map +1 -0
- package/dist/settings.d.ts +10 -249
- package/dist/settings.d.ts.map +1 -1
- package/dist/settings.js +5 -30
- package/dist/settings.js.map +1 -1
- package/dist/switchbotClient.d.ts +32 -0
- package/dist/switchbotClient.d.ts.map +1 -0
- package/dist/switchbotClient.js +259 -0
- package/dist/switchbotClient.js.map +1 -0
- package/dist/utils.d.ts +39 -50
- package/dist/utils.d.ts.map +1 -1
- package/dist/utils.js +39 -688
- package/dist/utils.js.map +1 -1
- package/docs/assets/highlight.css +14 -0
- package/docs/assets/icons.js +1 -1
- package/docs/assets/icons.svg +1 -1
- package/docs/assets/main.js +2 -2
- package/docs/assets/style.css +3 -3
- package/docs/index.html +77 -13
- package/docs/variables/default.html +1 -1
- package/package.json +22 -21
- package/scripts/e2e/README.md +25 -0
- package/scripts/e2e/curtain-e2e.sh +70 -0
- package/scripts/e2e/fan-e2e.sh +75 -0
- package/scripts/e2e/light-advanced-e2e.sh +97 -0
- package/scripts/e2e/light-e2e.sh +75 -0
- package/scripts/e2e/list-accessories.sh +19 -0
- package/scripts/e2e/lock-e2e.sh +65 -0
- package/scripts/generate-matter-maps.js +60 -0
- package/scripts/run-e2e-local.sh +14 -0
- package/src/deviceFactory.ts +122 -0
- package/src/devices/deviceBase.ts +141 -0
- package/src/devices/genericDevice.ts +965 -0
- package/src/homebridge-ui/public/index.html +77 -245
- package/src/homebridge-ui/server.ts +49 -41
- package/src/index.ts +5 -13
- package/src/platform.ts +395 -0
- package/src/settings.ts +12 -277
- package/src/switchbotClient.ts +266 -0
- package/src/utils.ts +45 -713
- package/test/accessory-restore.spec.ts +73 -0
- package/test/device-mapping.spec.ts +37 -0
- package/test/deviceFactory.spec.ts +18 -0
- package/test/e2e/run-e2e.spec.ts +50 -0
- package/test/fan-swing.spec.ts +29 -0
- package/test/helpers/matter-harness.ts +53 -0
- package/test/lock-users.spec.ts +44 -0
- package/test/matter-childbridge.spec.ts +55 -0
- package/test/matter-descriptors.spec.ts +97 -0
- package/test/matter-device-state.spec.ts +101 -0
- package/test/matter-integration.spec.ts +70 -0
- package/test/platform.integration.spec.ts +55 -0
- package/test/switchbot-client-debounce.spec.ts +131 -0
- package/test/switchbot-client-openapi.spec.ts +56 -0
- package/test/switchbotClient.spec.ts +10 -0
- package/test/utils.spec.ts +20 -0
- package/vitest.config.ts +7 -0
- package/coverage/base.css +0 -224
- package/coverage/block-navigation.js +0 -87
- package/coverage/clover.xml +0 -15847
- package/coverage/coverage-final.json +0 -42
- package/coverage/docs/assets/dmt/dmt-component-data.js.html +0 -85
- package/coverage/docs/assets/dmt/dmt-components.js.html +0 -286
- package/coverage/docs/assets/dmt/index.html +0 -131
- package/coverage/docs/assets/hierarchy.js.html +0 -85
- package/coverage/docs/assets/icons.js.html +0 -136
- package/coverage/docs/assets/index.html +0 -146
- package/coverage/docs/assets/main.js.html +0 -265
- package/coverage/favicon.png +0 -0
- package/coverage/index.html +0 -191
- package/coverage/prettify.css +0 -1
- package/coverage/prettify.js +0 -2
- package/coverage/sort-arrow-sprite.png +0 -0
- package/coverage/sorter.js +0 -196
- package/coverage/src/device/blindtilt.ts.html +0 -3238
- package/coverage/src/device/bot.ts.html +0 -2803
- package/coverage/src/device/ceilinglight.ts.html +0 -2338
- package/coverage/src/device/colorbulb.ts.html +0 -2824
- package/coverage/src/device/contact.ts.html +0 -1465
- package/coverage/src/device/curtain.ts.html +0 -2869
- package/coverage/src/device/device.ts.html +0 -2500
- package/coverage/src/device/fan.ts.html +0 -2242
- package/coverage/src/device/hub.ts.html +0 -1408
- package/coverage/src/device/humidifier.ts.html +0 -2116
- package/coverage/src/device/index.html +0 -416
- package/coverage/src/device/iosensor.ts.html +0 -1375
- package/coverage/src/device/lightstrip.ts.html +0 -2617
- package/coverage/src/device/lock.ts.html +0 -1963
- package/coverage/src/device/meter.ts.html +0 -1372
- package/coverage/src/device/meterplus.ts.html +0 -1384
- package/coverage/src/device/meterpro.ts.html +0 -1618
- package/coverage/src/device/motion.ts.html +0 -1264
- package/coverage/src/device/plug.ts.html +0 -1372
- package/coverage/src/device/relayswitch.ts.html +0 -2284
- package/coverage/src/device/robotvacuumcleaner.ts.html +0 -1810
- package/coverage/src/device/waterdetector.ts.html +0 -1294
- package/coverage/src/homebridge-ui/index.html +0 -116
- package/coverage/src/homebridge-ui/server.ts.html +0 -229
- package/coverage/src/index.html +0 -161
- package/coverage/src/index.ts.html +0 -124
- package/coverage/src/irdevice/airconditioner.ts.html +0 -1687
- package/coverage/src/irdevice/airpurifier.ts.html +0 -844
- package/coverage/src/irdevice/camera.ts.html +0 -475
- package/coverage/src/irdevice/fan.ts.html +0 -766
- package/coverage/src/irdevice/index.html +0 -251
- package/coverage/src/irdevice/irdevice.ts.html +0 -1117
- package/coverage/src/irdevice/light.ts.html +0 -826
- package/coverage/src/irdevice/other.ts.html +0 -2458
- package/coverage/src/irdevice/tv.ts.html +0 -1222
- package/coverage/src/irdevice/vacuumcleaner.ts.html +0 -466
- package/coverage/src/irdevice/waterheater.ts.html +0 -469
- package/coverage/src/platform.ts.html +0 -8776
- package/coverage/src/settings.ts.html +0 -934
- package/coverage/src/utils.ts.html +0 -2092
- package/dist/devices-hap/airpurifier.d.ts +0 -54
- package/dist/devices-hap/airpurifier.d.ts.map +0 -1
- package/dist/devices-hap/airpurifier.js +0 -527
- package/dist/devices-hap/airpurifier.js.map +0 -1
- package/dist/devices-hap/blindtilt.d.ts +0 -90
- package/dist/devices-hap/blindtilt.d.ts.map +0 -1
- package/dist/devices-hap/blindtilt.js +0 -974
- package/dist/devices-hap/blindtilt.js.map +0 -1
- package/dist/devices-hap/bot.d.ts +0 -102
- package/dist/devices-hap/bot.d.ts.map +0 -1
- package/dist/devices-hap/bot.js +0 -811
- package/dist/devices-hap/bot.js.map +0 -1
- package/dist/devices-hap/ceilinglight.d.ts +0 -85
- package/dist/devices-hap/ceilinglight.d.ts.map +0 -1
- package/dist/devices-hap/ceilinglight.js +0 -701
- package/dist/devices-hap/ceilinglight.js.map +0 -1
- package/dist/devices-hap/colorbulb.d.ts +0 -88
- package/dist/devices-hap/colorbulb.d.ts.map +0 -1
- package/dist/devices-hap/colorbulb.js +0 -881
- package/dist/devices-hap/colorbulb.js.map +0 -1
- package/dist/devices-hap/contact.d.ts +0 -44
- package/dist/devices-hap/contact.d.ts.map +0 -1
- package/dist/devices-hap/contact.js +0 -409
- package/dist/devices-hap/contact.js.map +0 -1
- package/dist/devices-hap/curtain.d.ts +0 -73
- package/dist/devices-hap/curtain.d.ts.map +0 -1
- package/dist/devices-hap/curtain.js +0 -869
- package/dist/devices-hap/curtain.js.map +0 -1
- package/dist/devices-hap/device.d.ts +0 -98
- package/dist/devices-hap/device.d.ts.map +0 -1
- package/dist/devices-hap/device.js +0 -749
- package/dist/devices-hap/device.js.map +0 -1
- package/dist/devices-hap/fan.d.ts +0 -69
- package/dist/devices-hap/fan.d.ts.map +0 -1
- package/dist/devices-hap/fan.js +0 -649
- package/dist/devices-hap/fan.js.map +0 -1
- package/dist/devices-hap/hub.d.ts +0 -37
- package/dist/devices-hap/hub.d.ts.map +0 -1
- package/dist/devices-hap/hub.js +0 -392
- package/dist/devices-hap/hub.js.map +0 -1
- package/dist/devices-hap/humidifier.d.ts +0 -68
- package/dist/devices-hap/humidifier.d.ts.map +0 -1
- package/dist/devices-hap/humidifier.js +0 -628
- package/dist/devices-hap/humidifier.js.map +0 -1
- package/dist/devices-hap/iosensor.d.ts +0 -42
- package/dist/devices-hap/iosensor.d.ts.map +0 -1
- package/dist/devices-hap/iosensor.js +0 -382
- package/dist/devices-hap/iosensor.js.map +0 -1
- package/dist/devices-hap/lightstrip.d.ts +0 -79
- package/dist/devices-hap/lightstrip.d.ts.map +0 -1
- package/dist/devices-hap/lightstrip.js +0 -797
- package/dist/devices-hap/lightstrip.js.map +0 -1
- package/dist/devices-hap/lock.d.ts +0 -53
- package/dist/devices-hap/lock.d.ts.map +0 -1
- package/dist/devices-hap/lock.js +0 -561
- package/dist/devices-hap/lock.js.map +0 -1
- package/dist/devices-hap/meter.d.ts +0 -37
- package/dist/devices-hap/meter.d.ts.map +0 -1
- package/dist/devices-hap/meter.js +0 -379
- package/dist/devices-hap/meter.js.map +0 -1
- package/dist/devices-hap/meterplus.d.ts +0 -42
- package/dist/devices-hap/meterplus.d.ts.map +0 -1
- package/dist/devices-hap/meterplus.js +0 -384
- package/dist/devices-hap/meterplus.js.map +0 -1
- package/dist/devices-hap/meterpro.d.ts +0 -43
- package/dist/devices-hap/meterpro.d.ts.map +0 -1
- package/dist/devices-hap/meterpro.js +0 -468
- package/dist/devices-hap/meterpro.js.map +0 -1
- package/dist/devices-hap/motion.d.ts +0 -42
- package/dist/devices-hap/motion.d.ts.map +0 -1
- package/dist/devices-hap/motion.js +0 -345
- package/dist/devices-hap/motion.js.map +0 -1
- package/dist/devices-hap/plug.d.ts +0 -49
- package/dist/devices-hap/plug.d.ts.map +0 -1
- package/dist/devices-hap/plug.js +0 -395
- package/dist/devices-hap/plug.js.map +0 -1
- package/dist/devices-hap/relayswitch.d.ts +0 -96
- package/dist/devices-hap/relayswitch.d.ts.map +0 -1
- package/dist/devices-hap/relayswitch.js +0 -642
- package/dist/devices-hap/relayswitch.js.map +0 -1
- package/dist/devices-hap/robotvacuumcleaner.d.ts +0 -54
- package/dist/devices-hap/robotvacuumcleaner.d.ts.map +0 -1
- package/dist/devices-hap/robotvacuumcleaner.js +0 -523
- package/dist/devices-hap/robotvacuumcleaner.js.map +0 -1
- package/dist/devices-hap/waterdetector.d.ts +0 -41
- package/dist/devices-hap/waterdetector.d.ts.map +0 -1
- package/dist/devices-hap/waterdetector.js +0 -356
- package/dist/devices-hap/waterdetector.js.map +0 -1
- package/dist/devices-matter/BaseMatterAccessory.d.ts +0 -63
- package/dist/devices-matter/BaseMatterAccessory.d.ts.map +0 -1
- package/dist/devices-matter/BaseMatterAccessory.js +0 -100
- package/dist/devices-matter/BaseMatterAccessory.js.map +0 -1
- package/dist/devices-matter/ColorLightAccessory.d.ts +0 -20
- package/dist/devices-matter/ColorLightAccessory.d.ts.map +0 -1
- package/dist/devices-matter/ColorLightAccessory.js +0 -95
- package/dist/devices-matter/ColorLightAccessory.js.map +0 -1
- package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts +0 -18
- package/dist/devices-matter/ColorTemperatureLightAccessory.d.ts.map +0 -1
- package/dist/devices-matter/ColorTemperatureLightAccessory.js +0 -78
- package/dist/devices-matter/ColorTemperatureLightAccessory.js.map +0 -1
- package/dist/devices-matter/ContactSensorAccessory.d.ts +0 -12
- package/dist/devices-matter/ContactSensorAccessory.d.ts.map +0 -1
- package/dist/devices-matter/ContactSensorAccessory.js +0 -34
- package/dist/devices-matter/ContactSensorAccessory.js.map +0 -1
- package/dist/devices-matter/DimmableLightAccessory.d.ts +0 -58
- package/dist/devices-matter/DimmableLightAccessory.d.ts.map +0 -1
- package/dist/devices-matter/DimmableLightAccessory.js +0 -167
- package/dist/devices-matter/DimmableLightAccessory.js.map +0 -1
- package/dist/devices-matter/DoorLockAccessory.d.ts +0 -14
- package/dist/devices-matter/DoorLockAccessory.d.ts.map +0 -1
- package/dist/devices-matter/DoorLockAccessory.js +0 -50
- package/dist/devices-matter/DoorLockAccessory.js.map +0 -1
- package/dist/devices-matter/ExtendedColorLightAccessory.d.ts +0 -21
- package/dist/devices-matter/ExtendedColorLightAccessory.d.ts.map +0 -1
- package/dist/devices-matter/ExtendedColorLightAccessory.js +0 -107
- package/dist/devices-matter/ExtendedColorLightAccessory.js.map +0 -1
- package/dist/devices-matter/FanAccessory.d.ts +0 -16
- package/dist/devices-matter/FanAccessory.d.ts.map +0 -1
- package/dist/devices-matter/FanAccessory.js +0 -81
- package/dist/devices-matter/FanAccessory.js.map +0 -1
- package/dist/devices-matter/HumiditySensorAccessory.d.ts +0 -12
- package/dist/devices-matter/HumiditySensorAccessory.d.ts.map +0 -1
- package/dist/devices-matter/HumiditySensorAccessory.js +0 -34
- package/dist/devices-matter/HumiditySensorAccessory.js.map +0 -1
- package/dist/devices-matter/LeakSensorAccessory.d.ts +0 -12
- package/dist/devices-matter/LeakSensorAccessory.d.ts.map +0 -1
- package/dist/devices-matter/LeakSensorAccessory.js +0 -33
- package/dist/devices-matter/LeakSensorAccessory.js.map +0 -1
- package/dist/devices-matter/LightSensorAccessory.d.ts +0 -12
- package/dist/devices-matter/LightSensorAccessory.d.ts.map +0 -1
- package/dist/devices-matter/LightSensorAccessory.js +0 -34
- package/dist/devices-matter/LightSensorAccessory.js.map +0 -1
- package/dist/devices-matter/OccupancySensorAccessory.d.ts +0 -12
- package/dist/devices-matter/OccupancySensorAccessory.d.ts.map +0 -1
- package/dist/devices-matter/OccupancySensorAccessory.js +0 -39
- package/dist/devices-matter/OccupancySensorAccessory.js.map +0 -1
- package/dist/devices-matter/OnOffLightAccessory.d.ts +0 -38
- package/dist/devices-matter/OnOffLightAccessory.d.ts.map +0 -1
- package/dist/devices-matter/OnOffLightAccessory.js +0 -118
- package/dist/devices-matter/OnOffLightAccessory.js.map +0 -1
- package/dist/devices-matter/OnOffOutletAccessory.d.ts +0 -12
- package/dist/devices-matter/OnOffOutletAccessory.d.ts.map +0 -1
- package/dist/devices-matter/OnOffOutletAccessory.js +0 -40
- package/dist/devices-matter/OnOffOutletAccessory.js.map +0 -1
- package/dist/devices-matter/OnOffSwitchAccessory.d.ts +0 -14
- package/dist/devices-matter/OnOffSwitchAccessory.d.ts.map +0 -1
- package/dist/devices-matter/OnOffSwitchAccessory.js +0 -42
- package/dist/devices-matter/OnOffSwitchAccessory.js.map +0 -1
- package/dist/devices-matter/RoboticVacuumAccessory.d.ts +0 -68
- package/dist/devices-matter/RoboticVacuumAccessory.d.ts.map +0 -1
- package/dist/devices-matter/RoboticVacuumAccessory.js +0 -334
- package/dist/devices-matter/RoboticVacuumAccessory.js.map +0 -1
- package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts +0 -11
- package/dist/devices-matter/SmokeCOAlarmAccessory.d.ts.map +0 -1
- package/dist/devices-matter/SmokeCOAlarmAccessory.js +0 -49
- package/dist/devices-matter/SmokeCOAlarmAccessory.js.map +0 -1
- package/dist/devices-matter/TemperatureSensorAccessory.d.ts +0 -12
- package/dist/devices-matter/TemperatureSensorAccessory.d.ts.map +0 -1
- package/dist/devices-matter/TemperatureSensorAccessory.js +0 -36
- package/dist/devices-matter/TemperatureSensorAccessory.js.map +0 -1
- package/dist/devices-matter/ThermostatAccessory.d.ts +0 -19
- package/dist/devices-matter/ThermostatAccessory.d.ts.map +0 -1
- package/dist/devices-matter/ThermostatAccessory.js +0 -95
- package/dist/devices-matter/ThermostatAccessory.js.map +0 -1
- package/dist/devices-matter/VenetianBlindAccessory.d.ts +0 -19
- package/dist/devices-matter/VenetianBlindAccessory.d.ts.map +0 -1
- package/dist/devices-matter/VenetianBlindAccessory.js +0 -99
- package/dist/devices-matter/VenetianBlindAccessory.js.map +0 -1
- package/dist/devices-matter/WindowBlindAccessory.d.ts +0 -17
- package/dist/devices-matter/WindowBlindAccessory.d.ts.map +0 -1
- package/dist/devices-matter/WindowBlindAccessory.js +0 -80
- package/dist/devices-matter/WindowBlindAccessory.js.map +0 -1
- package/dist/devices-matter/custom/PowerStripAccessory.d.ts +0 -97
- package/dist/devices-matter/custom/PowerStripAccessory.d.ts.map +0 -1
- package/dist/devices-matter/custom/PowerStripAccessory.js +0 -265
- package/dist/devices-matter/custom/PowerStripAccessory.js.map +0 -1
- package/dist/devices-matter/custom/index.d.ts +0 -8
- package/dist/devices-matter/custom/index.d.ts.map +0 -1
- package/dist/devices-matter/custom/index.js +0 -8
- package/dist/devices-matter/custom/index.js.map +0 -1
- package/dist/devices-matter/index.d.ts +0 -29
- package/dist/devices-matter/index.d.ts.map +0 -1
- package/dist/devices-matter/index.js +0 -28
- package/dist/devices-matter/index.js.map +0 -1
- package/dist/index.test.d.ts +0 -2
- package/dist/index.test.d.ts.map +0 -1
- package/dist/index.test.js +0 -14
- package/dist/index.test.js.map +0 -1
- package/dist/irdevice/airconditioner.d.ts +0 -61
- package/dist/irdevice/airconditioner.d.ts.map +0 -1
- package/dist/irdevice/airconditioner.js +0 -472
- package/dist/irdevice/airconditioner.js.map +0 -1
- package/dist/irdevice/airpurifier.d.ts +0 -50
- package/dist/irdevice/airpurifier.d.ts.map +0 -1
- package/dist/irdevice/airpurifier.js +0 -213
- package/dist/irdevice/airpurifier.js.map +0 -1
- package/dist/irdevice/camera.d.ts +0 -32
- package/dist/irdevice/camera.d.ts.map +0 -1
- package/dist/irdevice/camera.js +0 -107
- package/dist/irdevice/camera.js.map +0 -1
- package/dist/irdevice/fan.d.ts +0 -36
- package/dist/irdevice/fan.d.ts.map +0 -1
- package/dist/irdevice/fan.js +0 -200
- package/dist/irdevice/fan.js.map +0 -1
- package/dist/irdevice/irdevice.d.ts +0 -68
- package/dist/irdevice/irdevice.d.ts.map +0 -1
- package/dist/irdevice/irdevice.js +0 -298
- package/dist/irdevice/irdevice.js.map +0 -1
- package/dist/irdevice/light.d.ts +0 -36
- package/dist/irdevice/light.d.ts.map +0 -1
- package/dist/irdevice/light.js +0 -206
- package/dist/irdevice/light.js.map +0 -1
- package/dist/irdevice/other.d.ts +0 -57
- package/dist/irdevice/other.d.ts.map +0 -1
- package/dist/irdevice/other.js +0 -778
- package/dist/irdevice/other.js.map +0 -1
- package/dist/irdevice/tv.d.ts +0 -45
- package/dist/irdevice/tv.d.ts.map +0 -1
- package/dist/irdevice/tv.js +0 -327
- package/dist/irdevice/tv.js.map +0 -1
- package/dist/irdevice/vacuumcleaner.d.ts +0 -28
- package/dist/irdevice/vacuumcleaner.d.ts.map +0 -1
- package/dist/irdevice/vacuumcleaner.js +0 -104
- package/dist/irdevice/vacuumcleaner.js.map +0 -1
- package/dist/irdevice/waterheater.d.ts +0 -30
- package/dist/irdevice/waterheater.d.ts.map +0 -1
- package/dist/irdevice/waterheater.js +0 -105
- package/dist/irdevice/waterheater.js.map +0 -1
- package/dist/platform-hap.d.ts +0 -149
- package/dist/platform-hap.d.ts.map +0 -1
- package/dist/platform-hap.js +0 -2861
- package/dist/platform-hap.js.map +0 -1
- package/dist/platform-matter.d.ts +0 -101
- package/dist/platform-matter.d.ts.map +0 -1
- package/dist/platform-matter.js +0 -896
- package/dist/platform-matter.js.map +0 -1
- package/dist/verifyconfig.test.d.ts +0 -2
- package/dist/verifyconfig.test.d.ts.map +0 -1
- package/dist/verifyconfig.test.js +0 -167
- package/dist/verifyconfig.test.js.map +0 -1
- package/src/custom.d.ts +0 -7
- package/src/devices-hap/airpurifier.ts +0 -563
- package/src/devices-hap/blindtilt.ts +0 -1049
- package/src/devices-hap/bot.ts +0 -900
- package/src/devices-hap/ceilinglight.ts +0 -742
- package/src/devices-hap/colorbulb.ts +0 -904
- package/src/devices-hap/contact.ts +0 -457
- package/src/devices-hap/curtain.ts +0 -944
- package/src/devices-hap/device.ts +0 -811
- package/src/devices-hap/fan.ts +0 -711
- package/src/devices-hap/hub.ts +0 -439
- package/src/devices-hap/humidifier.ts +0 -669
- package/src/devices-hap/iosensor.ts +0 -427
- package/src/devices-hap/lightstrip.ts +0 -836
- package/src/devices-hap/lock.ts +0 -620
- package/src/devices-hap/meter.ts +0 -426
- package/src/devices-hap/meterplus.ts +0 -430
- package/src/devices-hap/meterpro.ts +0 -522
- package/src/devices-hap/motion.ts +0 -390
- package/src/devices-hap/plug.ts +0 -423
- package/src/devices-hap/relayswitch.ts +0 -727
- package/src/devices-hap/robotvacuumcleaner.ts +0 -568
- package/src/devices-hap/waterdetector.ts +0 -400
- package/src/devices-matter/BaseMatterAccessory.ts +0 -131
- package/src/devices-matter/ColorLightAccessory.ts +0 -110
- package/src/devices-matter/ColorTemperatureLightAccessory.ts +0 -92
- package/src/devices-matter/ContactSensorAccessory.ts +0 -41
- package/src/devices-matter/DimmableLightAccessory.ts +0 -192
- package/src/devices-matter/DoorLockAccessory.ts +0 -60
- package/src/devices-matter/ExtendedColorLightAccessory.ts +0 -123
- package/src/devices-matter/FanAccessory.ts +0 -95
- package/src/devices-matter/HumiditySensorAccessory.ts +0 -41
- package/src/devices-matter/LeakSensorAccessory.ts +0 -40
- package/src/devices-matter/LightSensorAccessory.ts +0 -41
- package/src/devices-matter/OccupancySensorAccessory.ts +0 -48
- package/src/devices-matter/OnOffLightAccessory.ts +0 -133
- package/src/devices-matter/OnOffOutletAccessory.ts +0 -46
- package/src/devices-matter/OnOffSwitchAccessory.ts +0 -51
- package/src/devices-matter/RoboticVacuumAccessory.ts +0 -407
- package/src/devices-matter/SmokeCOAlarmAccessory.ts +0 -59
- package/src/devices-matter/TemperatureSensorAccessory.ts +0 -43
- package/src/devices-matter/ThermostatAccessory.ts +0 -110
- package/src/devices-matter/VenetianBlindAccessory.ts +0 -115
- package/src/devices-matter/WindowBlindAccessory.ts +0 -92
- package/src/devices-matter/custom/PowerStripAccessory.ts +0 -309
- package/src/devices-matter/custom/index.ts +0 -8
- package/src/devices-matter/index.ts +0 -29
- package/src/index.test.ts +0 -19
- package/src/irdevice/airconditioner.ts +0 -533
- package/src/irdevice/airpurifier.ts +0 -252
- package/src/irdevice/camera.ts +0 -129
- package/src/irdevice/fan.ts +0 -226
- package/src/irdevice/irdevice.ts +0 -344
- package/src/irdevice/light.ts +0 -246
- package/src/irdevice/other.ts +0 -790
- package/src/irdevice/tv.ts +0 -378
- package/src/irdevice/vacuumcleaner.ts +0 -126
- package/src/irdevice/waterheater.ts +0 -127
- package/src/platform-hap.ts +0 -2997
- package/src/platform-matter.ts +0 -1010
- package/src/verifyconfig.test.ts +0 -197
|
@@ -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
|
-
}
|