Add support for scanner entities

This commit is contained in:
Erik
2026-04-14 16:55:16 +02:00
parent deba621779
commit 4eccc2d754
2 changed files with 64 additions and 19 deletions

View File

@@ -2,6 +2,7 @@
import voluptuous as vol
from homeassistant.components.zone import ENTITY_ID_HOME as ENTITY_ID_HOME_ZONE
from homeassistant.const import (
CONF_OPTIONS,
CONF_ZONE,
@@ -54,9 +55,18 @@ class ZoneTriggerBase(EntityTriggerBase):
return state.state not in (STATE_UNAVAILABLE, STATE_UNKNOWN)
def _in_target_zones(self, state: State) -> bool:
"""Check if the device is in any of the selected zones."""
in_zones = set(self._get_tracked_value(state) or [])
return bool(in_zones.intersection(self._zones))
"""Check if the device is in any of the selected zones.
For GPS-based trackers, uses the in_zones attribute.
For scanner-based trackers (no in_zones attribute), infers from
state: 'home' means the device is in zone.home.
"""
if (in_zones := self._get_tracked_value(state)) is not None:
return bool(set(in_zones).intersection(self._zones))
# Scanner tracker: state 'home' means in zone.home
if state.state == STATE_HOME:
return ENTITY_ID_HOME_ZONE in self._zones
return False
class EnteredZoneTrigger(ZoneTriggerBase):

View File

@@ -32,28 +32,29 @@ from tests.components.common import (
STATE_WORK_ZONE = "work"
def _dt_state(state: str, in_zones: list[str]) -> tuple[str, dict[str, list[str]]]:
"""Create a device tracker state tuple with in_zones attribute."""
def _gps_state(state: str, in_zones: list[str]) -> tuple[str, dict[str, list[str]]]:
"""Create a GPS-based device tracker state with in_zones attribute."""
return (state, {ATTR_IN_ZONES: in_zones})
ZONE_TRIGGERS = [
# Zone triggers for GPS-based trackers (have in_zones attribute)
GPS_ZONE_TRIGGERS = [
*parametrize_trigger_states(
trigger="device_tracker.entered_zone",
trigger_options={CONF_ZONE: ["zone.home", "zone.work"]},
target_states=[
# In zone.home
_dt_state(STATE_HOME, ["zone.home"]),
_gps_state(STATE_HOME, ["zone.home"]),
# In zone.work
_dt_state(STATE_WORK_ZONE, ["zone.work"]),
_gps_state(STATE_WORK_ZONE, ["zone.work"]),
# In both zones
_dt_state(STATE_HOME, ["zone.home", "zone.work"]),
_gps_state(STATE_HOME, ["zone.home", "zone.work"]),
],
other_states=[
# Not in any selected zone
_dt_state(STATE_NOT_HOME, []),
_gps_state(STATE_NOT_HOME, []),
# In an unrelated zone
_dt_state("school", ["zone.school"]),
_gps_state("school", ["zone.school"]),
],
),
*parametrize_trigger_states(
@@ -61,17 +62,42 @@ ZONE_TRIGGERS = [
trigger_options={CONF_ZONE: ["zone.home", "zone.work"]},
target_states=[
# Not in any selected zone
_dt_state(STATE_NOT_HOME, []),
_gps_state(STATE_NOT_HOME, []),
# In an unrelated zone only
_dt_state("school", ["zone.school"]),
_gps_state("school", ["zone.school"]),
],
other_states=[
# In zone.home
_dt_state(STATE_HOME, ["zone.home"]),
_gps_state(STATE_HOME, ["zone.home"]),
# In zone.work
_dt_state(STATE_WORK_ZONE, ["zone.work"]),
_gps_state(STATE_WORK_ZONE, ["zone.work"]),
# In both zones
_dt_state(STATE_HOME, ["zone.home", "zone.work"]),
_gps_state(STATE_HOME, ["zone.home", "zone.work"]),
],
),
]
# Zone triggers for scanner-based trackers (no in_zones attribute,
# state is 'home' or 'not_home')
SCANNER_ZONE_TRIGGERS = [
*parametrize_trigger_states(
trigger="device_tracker.entered_zone",
trigger_options={CONF_ZONE: ["zone.home"]},
target_states=[
STATE_HOME,
],
other_states=[
STATE_NOT_HOME,
],
),
*parametrize_trigger_states(
trigger="device_tracker.left_zone",
trigger_options={CONF_ZONE: ["zone.home"]},
target_states=[
STATE_NOT_HOME,
],
other_states=[
STATE_HOME,
],
),
]
@@ -292,7 +318,10 @@ async def test_device_tracker_zone_trigger_validation(
("trigger_target_config", "entity_id", "entities_in_target"),
parametrize_target_entities("device_tracker"),
)
@pytest.mark.parametrize(("trigger", "trigger_options", "states"), ZONE_TRIGGERS)
@pytest.mark.parametrize(
("trigger", "trigger_options", "states"),
[*GPS_ZONE_TRIGGERS, *SCANNER_ZONE_TRIGGERS],
)
async def test_device_tracker_zone_trigger_behavior_any(
hass: HomeAssistant,
target_device_trackers: dict[str, list[str]],
@@ -321,7 +350,10 @@ async def test_device_tracker_zone_trigger_behavior_any(
("trigger_target_config", "entity_id", "entities_in_target"),
parametrize_target_entities("device_tracker"),
)
@pytest.mark.parametrize(("trigger", "trigger_options", "states"), ZONE_TRIGGERS)
@pytest.mark.parametrize(
("trigger", "trigger_options", "states"),
[*GPS_ZONE_TRIGGERS, *SCANNER_ZONE_TRIGGERS],
)
async def test_device_tracker_zone_trigger_behavior_first(
hass: HomeAssistant,
target_device_trackers: dict[str, list[str]],
@@ -350,7 +382,10 @@ async def test_device_tracker_zone_trigger_behavior_first(
("trigger_target_config", "entity_id", "entities_in_target"),
parametrize_target_entities("device_tracker"),
)
@pytest.mark.parametrize(("trigger", "trigger_options", "states"), ZONE_TRIGGERS)
@pytest.mark.parametrize(
("trigger", "trigger_options", "states"),
[*GPS_ZONE_TRIGGERS, *SCANNER_ZONE_TRIGGERS],
)
async def test_device_tracker_zone_trigger_behavior_last(
hass: HomeAssistant,
target_device_trackers: dict[str, list[str]],