2019-12-01 22:12:57 +00:00
|
|
|
"""Intents for the light integration."""
|
|
|
|
import voluptuous as vol
|
|
|
|
|
2022-01-05 17:21:20 +00:00
|
|
|
from homeassistant.const import ATTR_ENTITY_ID, SERVICE_TURN_ON
|
2021-05-14 21:23:29 +00:00
|
|
|
from homeassistant.core import HomeAssistant, State
|
2019-12-01 22:12:57 +00:00
|
|
|
from homeassistant.helpers import intent
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2019-12-08 17:16:23 +00:00
|
|
|
import homeassistant.util.color as color_util
|
2019-12-01 22:12:57 +00:00
|
|
|
|
|
|
|
from . import (
|
2019-12-08 17:16:23 +00:00
|
|
|
ATTR_BRIGHTNESS_PCT,
|
2019-12-01 22:12:57 +00:00
|
|
|
ATTR_RGB_COLOR,
|
2021-05-14 21:23:29 +00:00
|
|
|
ATTR_SUPPORTED_COLOR_MODES,
|
2019-12-01 22:12:57 +00:00
|
|
|
DOMAIN,
|
2021-05-14 21:23:29 +00:00
|
|
|
brightness_supported,
|
|
|
|
color_supported,
|
2019-12-01 22:12:57 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
INTENT_SET = "HassLightSet"
|
|
|
|
|
|
|
|
|
|
|
|
async def async_setup_intents(hass: HomeAssistant) -> None:
|
|
|
|
"""Set up the light intents."""
|
2022-05-17 18:33:51 +00:00
|
|
|
intent.async_register(hass, SetIntentHandler())
|
2019-12-01 22:12:57 +00:00
|
|
|
|
|
|
|
|
2021-05-14 21:23:29 +00:00
|
|
|
def _test_supports_color(state: State) -> None:
|
|
|
|
"""Test if state supports colors."""
|
|
|
|
supported_color_modes = state.attributes.get(ATTR_SUPPORTED_COLOR_MODES)
|
|
|
|
if not color_supported(supported_color_modes):
|
|
|
|
raise intent.IntentHandleError(
|
|
|
|
f"Entity {state.name} does not support changing colors"
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
def _test_supports_brightness(state: State) -> None:
|
|
|
|
"""Test if state supports brightness."""
|
|
|
|
supported_color_modes = state.attributes.get(ATTR_SUPPORTED_COLOR_MODES)
|
|
|
|
if not brightness_supported(supported_color_modes):
|
|
|
|
raise intent.IntentHandleError(
|
|
|
|
f"Entity {state.name} does not support changing brightness"
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2019-12-01 22:12:57 +00:00
|
|
|
class SetIntentHandler(intent.IntentHandler):
|
|
|
|
"""Handle set color intents."""
|
|
|
|
|
|
|
|
intent_type = INTENT_SET
|
|
|
|
slot_schema = {
|
|
|
|
vol.Required("name"): cv.string,
|
|
|
|
vol.Optional("color"): color_util.color_name_to_rgb,
|
|
|
|
vol.Optional("brightness"): vol.All(vol.Coerce(int), vol.Range(0, 100)),
|
|
|
|
}
|
|
|
|
|
|
|
|
async def async_handle(self, intent_obj: intent.Intent) -> intent.IntentResponse:
|
|
|
|
"""Handle the hass intent."""
|
|
|
|
hass = intent_obj.hass
|
|
|
|
slots = self.async_validate_slots(intent_obj.slots)
|
2022-05-17 18:33:51 +00:00
|
|
|
state = intent.async_match_state(
|
|
|
|
hass, slots["name"]["value"], hass.states.async_all(DOMAIN)
|
2019-12-01 22:12:57 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
service_data = {ATTR_ENTITY_ID: state.entity_id}
|
|
|
|
speech_parts = []
|
|
|
|
|
|
|
|
if "color" in slots:
|
2021-05-14 21:23:29 +00:00
|
|
|
_test_supports_color(state)
|
2019-12-01 22:12:57 +00:00
|
|
|
service_data[ATTR_RGB_COLOR] = slots["color"]["value"]
|
|
|
|
# Use original passed in value of the color because we don't have
|
|
|
|
# human readable names for that internally.
|
2020-02-28 11:39:29 +00:00
|
|
|
speech_parts.append(f"the color {intent_obj.slots['color']['value']}")
|
2019-12-01 22:12:57 +00:00
|
|
|
|
|
|
|
if "brightness" in slots:
|
2021-05-14 21:23:29 +00:00
|
|
|
_test_supports_brightness(state)
|
2019-12-01 22:12:57 +00:00
|
|
|
service_data[ATTR_BRIGHTNESS_PCT] = slots["brightness"]["value"]
|
2020-02-28 11:39:29 +00:00
|
|
|
speech_parts.append(f"{slots['brightness']['value']}% brightness")
|
2019-12-01 22:12:57 +00:00
|
|
|
|
|
|
|
await hass.services.async_call(
|
|
|
|
DOMAIN, SERVICE_TURN_ON, service_data, context=intent_obj.context
|
|
|
|
)
|
|
|
|
|
|
|
|
response = intent_obj.create_response()
|
|
|
|
|
|
|
|
if not speech_parts: # No attributes changed
|
|
|
|
speech = f"Turned on {state.name}"
|
|
|
|
else:
|
|
|
|
parts = [f"Changed {state.name} to"]
|
|
|
|
for index, part in enumerate(speech_parts):
|
|
|
|
if index == 0:
|
|
|
|
parts.append(f" {part}")
|
|
|
|
elif index != len(speech_parts) - 1:
|
|
|
|
parts.append(f", {part}")
|
|
|
|
else:
|
|
|
|
parts.append(f" and {part}")
|
|
|
|
speech = "".join(parts)
|
|
|
|
|
|
|
|
response.async_set_speech(speech)
|
|
|
|
return response
|