bulb.py 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. from __future__ import annotations
  2. import logging
  3. from typing import Any
  4. from ..const.light import BulbColorMode, ColorMode
  5. from .base_light import SwitchbotSequenceBaseLight
  6. from .device import REQ_HEADER, SwitchbotOperationError, update_after_operation
  7. BULB_COMMAND_HEADER = "4701"
  8. BULB_REQUEST = f"{REQ_HEADER}4801"
  9. BULB_COMMAND = f"{REQ_HEADER}{BULB_COMMAND_HEADER}"
  10. # Bulb keys
  11. BULB_ON_KEY = f"{BULB_COMMAND}01"
  12. BULB_OFF_KEY = f"{BULB_COMMAND}02"
  13. RGB_BRIGHTNESS_KEY = f"{BULB_COMMAND}12"
  14. CW_BRIGHTNESS_KEY = f"{BULB_COMMAND}13"
  15. BRIGHTNESS_KEY = f"{BULB_COMMAND}14"
  16. RGB_KEY = f"{BULB_COMMAND}16"
  17. CW_KEY = f"{BULB_COMMAND}17"
  18. DEVICE_GET_VERSION_KEY = "570003"
  19. DEVICE_GET_BASIC_SETTINGS_KEY = "570f4801"
  20. _LOGGER = logging.getLogger(__name__)
  21. EFFECT_DICT = {
  22. "Colorful": "570F4701010300",
  23. "Flickering": "570F4701010301",
  24. "Breathing": "570F4701010302",
  25. }
  26. # Private mapping from device-specific color modes to original ColorMode enum
  27. _BULB_COLOR_MODE_MAP = {
  28. BulbColorMode.COLOR_TEMP: ColorMode.COLOR_TEMP,
  29. BulbColorMode.RGB: ColorMode.RGB,
  30. BulbColorMode.DYNAMIC: ColorMode.EFFECT,
  31. BulbColorMode.UNKNOWN: ColorMode.OFF,
  32. }
  33. class SwitchbotBulb(SwitchbotSequenceBaseLight):
  34. """Representation of a Switchbot bulb."""
  35. @property
  36. def color_modes(self) -> set[ColorMode]:
  37. """Return the supported color modes."""
  38. return {ColorMode.RGB, ColorMode.COLOR_TEMP}
  39. @property
  40. def color_mode(self) -> ColorMode:
  41. """Return the current color mode."""
  42. device_mode = BulbColorMode(self._get_adv_value("color_mode") or 10)
  43. return _BULB_COLOR_MODE_MAP.get(device_mode, ColorMode.OFF)
  44. @property
  45. def get_effect_list(self) -> list[str]:
  46. """Return the list of supported effects."""
  47. return list(EFFECT_DICT.keys())
  48. @update_after_operation
  49. async def turn_on(self) -> bool:
  50. """Turn device on."""
  51. result = await self._send_command(BULB_ON_KEY)
  52. return self._check_command_result(result, 0, {1})
  53. @update_after_operation
  54. async def turn_off(self) -> bool:
  55. """Turn device off."""
  56. result = await self._send_command(BULB_OFF_KEY)
  57. return self._check_command_result(result, 0, {1})
  58. @update_after_operation
  59. async def set_brightness(self, brightness: int) -> bool:
  60. """Set brightness."""
  61. assert 0 <= brightness <= 100, "Brightness must be between 0 and 100"
  62. result = await self._send_command(f"{BRIGHTNESS_KEY}{brightness:02X}")
  63. return self._check_command_result(result, 0, {1})
  64. @update_after_operation
  65. async def set_color_temp(self, brightness: int, color_temp: int) -> bool:
  66. """Set color temp."""
  67. assert 0 <= brightness <= 100, "Brightness must be between 0 and 100"
  68. assert 2700 <= color_temp <= 6500, "Color Temp must be between 2700 and 6500"
  69. result = await self._send_command(
  70. f"{CW_BRIGHTNESS_KEY}{brightness:02X}{color_temp:04X}"
  71. )
  72. return self._check_command_result(result, 0, {1})
  73. @update_after_operation
  74. async def set_rgb(self, brightness: int, r: int, g: int, b: int) -> bool:
  75. """Set rgb."""
  76. assert 0 <= brightness <= 100, "Brightness must be between 0 and 100"
  77. assert 0 <= r <= 255, "r must be between 0 and 255"
  78. assert 0 <= g <= 255, "g must be between 0 and 255"
  79. assert 0 <= b <= 255, "b must be between 0 and 255"
  80. result = await self._send_command(
  81. f"{RGB_BRIGHTNESS_KEY}{brightness:02X}{r:02X}{g:02X}{b:02X}"
  82. )
  83. return self._check_command_result(result, 0, {1})
  84. @update_after_operation
  85. async def set_effect(self, effect: str) -> bool:
  86. """Set effect."""
  87. effect_template = EFFECT_DICT.get(effect)
  88. if not effect_template:
  89. raise SwitchbotOperationError(f"Effect {effect} not supported")
  90. result = await self._send_command(effect_template)
  91. if result:
  92. self._override_state({"effect": effect})
  93. return result
  94. async def get_basic_info(self) -> dict[str, Any] | None:
  95. """Get device basic settings."""
  96. if not (_data := await self._get_basic_info(DEVICE_GET_BASIC_SETTINGS_KEY)):
  97. return None
  98. if not (_version_info := await self._get_basic_info(DEVICE_GET_VERSION_KEY)):
  99. return None
  100. _LOGGER.debug(
  101. "data: %s, version info: %s, address: %s",
  102. _data,
  103. _version_info,
  104. self._device.address,
  105. )
  106. self._state["r"] = _data[3]
  107. self._state["g"] = _data[4]
  108. self._state["b"] = _data[5]
  109. self._state["cw"] = int.from_bytes(_data[6:8], "big")
  110. return {
  111. "isOn": bool(_data[1] & 0b10000000),
  112. "brightness": _data[2] & 0b01111111,
  113. "r": self._state["r"],
  114. "g": self._state["g"],
  115. "b": self._state["b"],
  116. "cw": self._state["cw"],
  117. "color_mode": _data[10] & 0b00001111,
  118. "firmware": _version_info[2] / 10.0,
  119. }