ceiling_light.py 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105
  1. from __future__ import annotations
  2. import logging
  3. from typing import Any
  4. from ..const.light import (
  5. DEFAULT_COLOR_TEMP,
  6. CeilingLightColorMode,
  7. ColorMode,
  8. )
  9. from .base_light import SwitchbotSequenceBaseLight
  10. from .device import REQ_HEADER, update_after_operation
  11. CEILING_LIGHT_COMMAND_HEADER = "5401"
  12. CEILING_LIGHT_REQUEST = f"{REQ_HEADER}5501"
  13. CEILING_LIGHT_COMMAND = f"{REQ_HEADER}{CEILING_LIGHT_COMMAND_HEADER}"
  14. CEILING_LIGHT_ON_KEY = f"{CEILING_LIGHT_COMMAND}01FF01FFFF"
  15. CEILING_LIGHT_OFF_KEY = f"{CEILING_LIGHT_COMMAND}02FF01FFFF"
  16. CW_BRIGHTNESS_KEY = f"{CEILING_LIGHT_COMMAND}010001"
  17. BRIGHTNESS_KEY = f"{CEILING_LIGHT_COMMAND}01FF01"
  18. DEVICE_GET_VERSION_KEY = "5702"
  19. DEVICE_GET_BASIC_SETTINGS_KEY = "570f5581"
  20. _LOGGER = logging.getLogger(__name__)
  21. # Private mapping from device-specific color modes to original ColorMode enum
  22. _CEILING_LIGHT_COLOR_MODE_MAP = {
  23. CeilingLightColorMode.COLOR_TEMP: ColorMode.COLOR_TEMP,
  24. CeilingLightColorMode.NIGHT: ColorMode.COLOR_TEMP,
  25. CeilingLightColorMode.MUSIC: ColorMode.EFFECT,
  26. CeilingLightColorMode.UNKNOWN: ColorMode.OFF,
  27. }
  28. class SwitchbotCeilingLight(SwitchbotSequenceBaseLight):
  29. """Representation of a Switchbot ceiling light."""
  30. @property
  31. def color_modes(self) -> set[ColorMode]:
  32. """Return the supported color modes."""
  33. return {ColorMode.COLOR_TEMP}
  34. @property
  35. def color_mode(self) -> ColorMode:
  36. """Return the current color mode."""
  37. device_mode = CeilingLightColorMode(self._get_adv_value("color_mode") or 10)
  38. return _CEILING_LIGHT_COLOR_MODE_MAP.get(device_mode, ColorMode.OFF)
  39. @update_after_operation
  40. async def turn_on(self) -> bool:
  41. """Turn device on."""
  42. result = await self._send_command(CEILING_LIGHT_ON_KEY)
  43. return self._check_command_result(result, 0, {1})
  44. @update_after_operation
  45. async def turn_off(self) -> bool:
  46. """Turn device off."""
  47. result = await self._send_command(CEILING_LIGHT_OFF_KEY)
  48. return self._check_command_result(result, 0, {1})
  49. @update_after_operation
  50. async def set_brightness(self, brightness: int) -> bool:
  51. """Set brightness."""
  52. assert 0 <= brightness <= 100, "Brightness must be between 0 and 100"
  53. color_temp = self._state.get("cw", DEFAULT_COLOR_TEMP)
  54. result = await self._send_command(
  55. f"{BRIGHTNESS_KEY}{brightness:02X}{color_temp:04X}"
  56. )
  57. return self._check_command_result(result, 0, {1})
  58. @update_after_operation
  59. async def set_color_temp(self, brightness: int, color_temp: int) -> bool:
  60. """Set color temp."""
  61. assert 0 <= brightness <= 100, "Brightness must be between 0 and 100"
  62. assert 2700 <= color_temp <= 6500, "Color Temp must be between 2700 and 6500"
  63. result = await self._send_command(
  64. f"{CW_BRIGHTNESS_KEY}{brightness:02X}{color_temp:04X}"
  65. )
  66. return self._check_command_result(result, 0, {1})
  67. async def get_basic_info(self) -> dict[str, Any] | None:
  68. """Get device basic settings."""
  69. if not (_data := await self._get_basic_info(DEVICE_GET_BASIC_SETTINGS_KEY)):
  70. return None
  71. if not (_version_info := await self._get_basic_info(DEVICE_GET_VERSION_KEY)):
  72. return None
  73. _LOGGER.debug(
  74. "data: %s, version info: %s, address: %s",
  75. _data,
  76. _version_info,
  77. self._device.address,
  78. )
  79. self._state["cw"] = int.from_bytes(_data[3:5], "big")
  80. return {
  81. "isOn": bool(_data[1] & 0b10000000),
  82. "color_mode": _data[1] & 0b01000000,
  83. "brightness": _data[2] & 0b01111111,
  84. "cw": self._state["cw"],
  85. "firmware": _version_info[2] / 10.0,
  86. }