mirror of
https://github.com/home-assistant/core.git
synced 2025-07-23 13:17:32 +00:00
Support default transition in light profiles (#36747)
Co-authored-by: Paulus Schoutsen <paulus@home-assistant.io>
This commit is contained in:
parent
5cf7b1b1bc
commit
6e87c2ad3e
@ -123,7 +123,12 @@ LIGHT_TURN_ON_SCHEMA = {
|
|||||||
|
|
||||||
|
|
||||||
PROFILE_SCHEMA = vol.Schema(
|
PROFILE_SCHEMA = vol.Schema(
|
||||||
vol.ExactSequence((str, cv.small_float, cv.small_float, cv.byte))
|
vol.Any(
|
||||||
|
vol.ExactSequence((str, cv.small_float, cv.small_float, cv.byte)),
|
||||||
|
vol.ExactSequence(
|
||||||
|
(str, cv.small_float, cv.small_float, cv.byte, cv.positive_int)
|
||||||
|
),
|
||||||
|
)
|
||||||
)
|
)
|
||||||
|
|
||||||
_LOGGER = logging.getLogger(__name__)
|
_LOGGER = logging.getLogger(__name__)
|
||||||
@ -141,6 +146,8 @@ def preprocess_turn_on_alternatives(params):
|
|||||||
if profile is not None:
|
if profile is not None:
|
||||||
params.setdefault(ATTR_XY_COLOR, profile[:2])
|
params.setdefault(ATTR_XY_COLOR, profile[:2])
|
||||||
params.setdefault(ATTR_BRIGHTNESS, profile[2])
|
params.setdefault(ATTR_BRIGHTNESS, profile[2])
|
||||||
|
if len(profile) > 3:
|
||||||
|
params.setdefault(ATTR_TRANSITION, profile[3])
|
||||||
|
|
||||||
color_name = params.pop(ATTR_COLOR_NAME, None)
|
color_name = params.pop(ATTR_COLOR_NAME, None)
|
||||||
if color_name is not None:
|
if color_name is not None:
|
||||||
@ -313,8 +320,22 @@ class Profiles:
|
|||||||
|
|
||||||
try:
|
try:
|
||||||
for rec in reader:
|
for rec in reader:
|
||||||
profile, color_x, color_y, brightness = PROFILE_SCHEMA(rec)
|
(
|
||||||
profiles[profile] = (color_x, color_y, brightness)
|
profile,
|
||||||
|
color_x,
|
||||||
|
color_y,
|
||||||
|
brightness,
|
||||||
|
*transition,
|
||||||
|
) = PROFILE_SCHEMA(rec)
|
||||||
|
|
||||||
|
transition = transition[0] if transition else 0
|
||||||
|
|
||||||
|
profiles[profile] = (
|
||||||
|
color_x,
|
||||||
|
color_y,
|
||||||
|
brightness,
|
||||||
|
transition,
|
||||||
|
)
|
||||||
except vol.MultipleInvalid as ex:
|
except vol.MultipleInvalid as ex:
|
||||||
_LOGGER.error(
|
_LOGGER.error(
|
||||||
"Error parsing light profile from %s: %s", profile_path, ex
|
"Error parsing light profile from %s: %s", profile_path, ex
|
||||||
|
@ -243,12 +243,14 @@ class TestLight(unittest.TestCase):
|
|||||||
assert {} == data
|
assert {} == data
|
||||||
|
|
||||||
# One of the light profiles
|
# One of the light profiles
|
||||||
prof_name, prof_h, prof_s, prof_bri = "relax", 35.932, 69.412, 144
|
prof_name, prof_h, prof_s, prof_bri, prof_t = "relax", 35.932, 69.412, 144, 0
|
||||||
|
|
||||||
# Test light profiles
|
# Test light profiles
|
||||||
common.turn_on(self.hass, ent1.entity_id, profile=prof_name)
|
common.turn_on(self.hass, ent1.entity_id, profile=prof_name)
|
||||||
# Specify a profile and a brightness attribute to overwrite it
|
# Specify a profile and a brightness attribute to overwrite it
|
||||||
common.turn_on(self.hass, ent2.entity_id, profile=prof_name, brightness=100)
|
common.turn_on(
|
||||||
|
self.hass, ent2.entity_id, profile=prof_name, brightness=100, transition=1
|
||||||
|
)
|
||||||
|
|
||||||
self.hass.block_till_done()
|
self.hass.block_till_done()
|
||||||
|
|
||||||
@ -256,12 +258,14 @@ class TestLight(unittest.TestCase):
|
|||||||
assert {
|
assert {
|
||||||
light.ATTR_BRIGHTNESS: prof_bri,
|
light.ATTR_BRIGHTNESS: prof_bri,
|
||||||
light.ATTR_HS_COLOR: (prof_h, prof_s),
|
light.ATTR_HS_COLOR: (prof_h, prof_s),
|
||||||
|
light.ATTR_TRANSITION: prof_t,
|
||||||
} == data
|
} == data
|
||||||
|
|
||||||
_, data = ent2.last_call("turn_on")
|
_, data = ent2.last_call("turn_on")
|
||||||
assert {
|
assert {
|
||||||
light.ATTR_BRIGHTNESS: 100,
|
light.ATTR_BRIGHTNESS: 100,
|
||||||
light.ATTR_HS_COLOR: (prof_h, prof_s),
|
light.ATTR_HS_COLOR: (prof_h, prof_s),
|
||||||
|
light.ATTR_TRANSITION: 1,
|
||||||
} == data
|
} == data
|
||||||
|
|
||||||
# Test toggle with parameters
|
# Test toggle with parameters
|
||||||
@ -271,6 +275,7 @@ class TestLight(unittest.TestCase):
|
|||||||
assert {
|
assert {
|
||||||
light.ATTR_BRIGHTNESS: 255,
|
light.ATTR_BRIGHTNESS: 255,
|
||||||
light.ATTR_HS_COLOR: (prof_h, prof_s),
|
light.ATTR_HS_COLOR: (prof_h, prof_s),
|
||||||
|
light.ATTR_TRANSITION: prof_t,
|
||||||
} == data
|
} == data
|
||||||
|
|
||||||
# Test bad data
|
# Test bad data
|
||||||
@ -314,8 +319,8 @@ class TestLight(unittest.TestCase):
|
|||||||
|
|
||||||
# Setup a wrong light file
|
# Setup a wrong light file
|
||||||
with open(user_light_file, "w") as user_file:
|
with open(user_light_file, "w") as user_file:
|
||||||
user_file.write("id,x,y,brightness\n")
|
user_file.write("id,x,y,brightness,transition\n")
|
||||||
user_file.write("I,WILL,NOT,WORK\n")
|
user_file.write("I,WILL,NOT,WORK,EVER\n")
|
||||||
|
|
||||||
assert not setup_component(
|
assert not setup_component(
|
||||||
self.hass, light.DOMAIN, {light.DOMAIN: {CONF_PLATFORM: "test"}}
|
self.hass, light.DOMAIN, {light.DOMAIN: {CONF_PLATFORM: "test"}}
|
||||||
@ -347,7 +352,11 @@ class TestLight(unittest.TestCase):
|
|||||||
_, data = ent1.last_call("turn_on")
|
_, data = ent1.last_call("turn_on")
|
||||||
|
|
||||||
assert light.is_on(self.hass, ent1.entity_id)
|
assert light.is_on(self.hass, ent1.entity_id)
|
||||||
assert {light.ATTR_HS_COLOR: (71.059, 100), light.ATTR_BRIGHTNESS: 100} == data
|
assert {
|
||||||
|
light.ATTR_HS_COLOR: (71.059, 100),
|
||||||
|
light.ATTR_BRIGHTNESS: 100,
|
||||||
|
light.ATTR_TRANSITION: 0,
|
||||||
|
} == data
|
||||||
|
|
||||||
common.turn_on(self.hass, ent1.entity_id, profile="test_off")
|
common.turn_on(self.hass, ent1.entity_id, profile="test_off")
|
||||||
|
|
||||||
@ -356,7 +365,48 @@ class TestLight(unittest.TestCase):
|
|||||||
_, data = ent1.last_call("turn_off")
|
_, data = ent1.last_call("turn_off")
|
||||||
|
|
||||||
assert not light.is_on(self.hass, ent1.entity_id)
|
assert not light.is_on(self.hass, ent1.entity_id)
|
||||||
assert {} == data
|
assert {light.ATTR_TRANSITION: 0} == data
|
||||||
|
|
||||||
|
def test_light_profiles_with_transition(self):
|
||||||
|
"""Test light profiles with transition."""
|
||||||
|
platform = getattr(self.hass.components, "test.light")
|
||||||
|
platform.init()
|
||||||
|
|
||||||
|
user_light_file = self.hass.config.path(light.LIGHT_PROFILES_FILE)
|
||||||
|
|
||||||
|
with open(user_light_file, "w") as user_file:
|
||||||
|
user_file.write("id,x,y,brightness,transition\n")
|
||||||
|
user_file.write("test,.4,.6,100,2\n")
|
||||||
|
user_file.write("test_off,0,0,0,0\n")
|
||||||
|
|
||||||
|
assert setup_component(
|
||||||
|
self.hass, light.DOMAIN, {light.DOMAIN: {CONF_PLATFORM: "test"}}
|
||||||
|
)
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
ent1, _, _ = platform.ENTITIES
|
||||||
|
|
||||||
|
common.turn_on(self.hass, ent1.entity_id, profile="test")
|
||||||
|
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
_, data = ent1.last_call("turn_on")
|
||||||
|
|
||||||
|
assert light.is_on(self.hass, ent1.entity_id)
|
||||||
|
assert {
|
||||||
|
light.ATTR_HS_COLOR: (71.059, 100),
|
||||||
|
light.ATTR_BRIGHTNESS: 100,
|
||||||
|
light.ATTR_TRANSITION: 2,
|
||||||
|
} == data
|
||||||
|
|
||||||
|
common.turn_on(self.hass, ent1.entity_id, profile="test_off")
|
||||||
|
|
||||||
|
self.hass.block_till_done()
|
||||||
|
|
||||||
|
_, data = ent1.last_call("turn_off")
|
||||||
|
|
||||||
|
assert not light.is_on(self.hass, ent1.entity_id)
|
||||||
|
assert {light.ATTR_TRANSITION: 0} == data
|
||||||
|
|
||||||
def test_default_profiles_group(self):
|
def test_default_profiles_group(self):
|
||||||
"""Test default turn-on light profile for all lights."""
|
"""Test default turn-on light profile for all lights."""
|
||||||
@ -377,7 +427,9 @@ class TestLight(unittest.TestCase):
|
|||||||
return StringIO(profile_data)
|
return StringIO(profile_data)
|
||||||
return real_open(path, *args, **kwargs)
|
return real_open(path, *args, **kwargs)
|
||||||
|
|
||||||
profile_data = "id,x,y,brightness\ngroup.all_lights.default,.4,.6,99\n"
|
profile_data = (
|
||||||
|
"id,x,y,brightness,transition\ngroup.all_lights.default,.4,.6,99,2\n"
|
||||||
|
)
|
||||||
with mock.patch("os.path.isfile", side_effect=_mock_isfile), mock.patch(
|
with mock.patch("os.path.isfile", side_effect=_mock_isfile), mock.patch(
|
||||||
"builtins.open", side_effect=_mock_open
|
"builtins.open", side_effect=_mock_open
|
||||||
), mock_storage():
|
), mock_storage():
|
||||||
@ -390,7 +442,11 @@ class TestLight(unittest.TestCase):
|
|||||||
common.turn_on(self.hass, ent.entity_id)
|
common.turn_on(self.hass, ent.entity_id)
|
||||||
self.hass.block_till_done()
|
self.hass.block_till_done()
|
||||||
_, data = ent.last_call("turn_on")
|
_, data = ent.last_call("turn_on")
|
||||||
assert {light.ATTR_HS_COLOR: (71.059, 100), light.ATTR_BRIGHTNESS: 99} == data
|
assert {
|
||||||
|
light.ATTR_HS_COLOR: (71.059, 100),
|
||||||
|
light.ATTR_BRIGHTNESS: 99,
|
||||||
|
light.ATTR_TRANSITION: 2,
|
||||||
|
} == data
|
||||||
|
|
||||||
def test_default_profiles_light(self):
|
def test_default_profiles_light(self):
|
||||||
"""Test default turn-on light profile for a specific light."""
|
"""Test default turn-on light profile for a specific light."""
|
||||||
@ -412,9 +468,9 @@ class TestLight(unittest.TestCase):
|
|||||||
return real_open(path, *args, **kwargs)
|
return real_open(path, *args, **kwargs)
|
||||||
|
|
||||||
profile_data = (
|
profile_data = (
|
||||||
"id,x,y,brightness\n"
|
"id,x,y,brightness,transition\n"
|
||||||
+ "group.all_lights.default,.3,.5,200\n"
|
+ "group.all_lights.default,.3,.5,200,0\n"
|
||||||
+ "light.ceiling_2.default,.6,.6,100\n"
|
+ "light.ceiling_2.default,.6,.6,100,3\n"
|
||||||
)
|
)
|
||||||
with mock.patch("os.path.isfile", side_effect=_mock_isfile), mock.patch(
|
with mock.patch("os.path.isfile", side_effect=_mock_isfile), mock.patch(
|
||||||
"builtins.open", side_effect=_mock_open
|
"builtins.open", side_effect=_mock_open
|
||||||
@ -430,7 +486,11 @@ class TestLight(unittest.TestCase):
|
|||||||
common.turn_on(self.hass, dev.entity_id)
|
common.turn_on(self.hass, dev.entity_id)
|
||||||
self.hass.block_till_done()
|
self.hass.block_till_done()
|
||||||
_, data = dev.last_call("turn_on")
|
_, data = dev.last_call("turn_on")
|
||||||
assert {light.ATTR_HS_COLOR: (50.353, 100), light.ATTR_BRIGHTNESS: 100} == data
|
assert {
|
||||||
|
light.ATTR_HS_COLOR: (50.353, 100),
|
||||||
|
light.ATTR_BRIGHTNESS: 100,
|
||||||
|
light.ATTR_TRANSITION: 3,
|
||||||
|
} == data
|
||||||
|
|
||||||
|
|
||||||
async def test_light_context(hass, hass_admin_user):
|
async def test_light_context(hass, hass_admin_user):
|
||||||
|
Loading…
x
Reference in New Issue
Block a user