mirror of
https://github.com/home-assistant/core.git
synced 2025-07-27 23:27:37 +00:00
Enable polling for hardwired powerview devices (#73659)
* Enable polling for hardwired powerview devices * Update homeassistant/components/hunterdouglas_powerview/cover.py * Update homeassistant/components/hunterdouglas_powerview/cover.py * docs were wrong * Update homeassistant/components/hunterdouglas_powerview/cover.py * Update homeassistant/components/hunterdouglas_powerview/sensor.py
This commit is contained in:
parent
8e3d9d7435
commit
120479acef
@ -87,3 +87,9 @@ POS_KIND_PRIMARY = 1
|
|||||||
POS_KIND_SECONDARY = 2
|
POS_KIND_SECONDARY = 2
|
||||||
POS_KIND_VANE = 3
|
POS_KIND_VANE = 3
|
||||||
POS_KIND_ERROR = 4
|
POS_KIND_ERROR = 4
|
||||||
|
|
||||||
|
|
||||||
|
ATTR_BATTERY_KIND = "batteryKind"
|
||||||
|
BATTERY_KIND_HARDWIRED = 1
|
||||||
|
BATTERY_KIND_BATTERY = 2
|
||||||
|
BATTERY_KIND_RECHARGABLE = 3
|
||||||
|
@ -4,6 +4,7 @@ from __future__ import annotations
|
|||||||
import asyncio
|
import asyncio
|
||||||
from collections.abc import Iterable
|
from collections.abc import Iterable
|
||||||
from contextlib import suppress
|
from contextlib import suppress
|
||||||
|
from datetime import timedelta
|
||||||
import logging
|
import logging
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
@ -74,6 +75,8 @@ RESYNC_DELAY = 60
|
|||||||
# implemented for top/down shades, but also works fine with normal shades
|
# implemented for top/down shades, but also works fine with normal shades
|
||||||
CLOSED_POSITION = (0.75 / 100) * (MAX_POSITION - MIN_POSITION)
|
CLOSED_POSITION = (0.75 / 100) * (MAX_POSITION - MIN_POSITION)
|
||||||
|
|
||||||
|
SCAN_INTERVAL = timedelta(minutes=10)
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(
|
async def async_setup_entry(
|
||||||
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
hass: HomeAssistant, entry: ConfigEntry, async_add_entities: AddEntitiesCallback
|
||||||
@ -152,8 +155,6 @@ def hass_position_to_hd(hass_position: int, max_val: int = MAX_POSITION) -> int:
|
|||||||
class PowerViewShadeBase(ShadeEntity, CoverEntity):
|
class PowerViewShadeBase(ShadeEntity, CoverEntity):
|
||||||
"""Representation of a powerview shade."""
|
"""Representation of a powerview shade."""
|
||||||
|
|
||||||
# The hub frequently reports stale states
|
|
||||||
_attr_assumed_state = True
|
|
||||||
_attr_device_class = CoverDeviceClass.SHADE
|
_attr_device_class = CoverDeviceClass.SHADE
|
||||||
_attr_supported_features = 0
|
_attr_supported_features = 0
|
||||||
|
|
||||||
@ -174,6 +175,26 @@ class PowerViewShadeBase(ShadeEntity, CoverEntity):
|
|||||||
self._attr_supported_features |= CoverEntityFeature.STOP
|
self._attr_supported_features |= CoverEntityFeature.STOP
|
||||||
self._forced_resync = None
|
self._forced_resync = None
|
||||||
|
|
||||||
|
@property
|
||||||
|
def assumed_state(self) -> bool:
|
||||||
|
"""If the device is hard wired we are polling state.
|
||||||
|
|
||||||
|
The hub will frequently provide the wrong state
|
||||||
|
for battery power devices so we set assumed
|
||||||
|
state in this case.
|
||||||
|
"""
|
||||||
|
return not self._is_hard_wired
|
||||||
|
|
||||||
|
@property
|
||||||
|
def should_poll(self) -> bool:
|
||||||
|
"""Only poll if the device is hard wired.
|
||||||
|
|
||||||
|
We cannot poll battery powered devices
|
||||||
|
as it would drain their batteries in a matter
|
||||||
|
of days.
|
||||||
|
"""
|
||||||
|
return self._is_hard_wired
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def extra_state_attributes(self) -> dict[str, str]:
|
def extra_state_attributes(self) -> dict[str, str]:
|
||||||
"""Return the state attributes."""
|
"""Return the state attributes."""
|
||||||
@ -336,15 +357,29 @@ class PowerViewShadeBase(ShadeEntity, CoverEntity):
|
|||||||
"""Cancel any pending refreshes."""
|
"""Cancel any pending refreshes."""
|
||||||
self._async_cancel_scheduled_transition_update()
|
self._async_cancel_scheduled_transition_update()
|
||||||
|
|
||||||
|
@property
|
||||||
|
def _update_in_progress(self) -> bool:
|
||||||
|
"""Check if an update is already in progress."""
|
||||||
|
return bool(self._scheduled_transition_update or self._forced_resync)
|
||||||
|
|
||||||
@callback
|
@callback
|
||||||
def _async_update_shade_from_group(self) -> None:
|
def _async_update_shade_from_group(self) -> None:
|
||||||
"""Update with new data from the coordinator."""
|
"""Update with new data from the coordinator."""
|
||||||
if self._scheduled_transition_update or self._forced_resync:
|
if self._update_in_progress:
|
||||||
# If a transition is in progress the data will be wrong
|
# If a transition is in progress the data will be wrong
|
||||||
return
|
return
|
||||||
self.data.update_from_group_data(self._shade.id)
|
self.data.update_from_group_data(self._shade.id)
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
||||||
|
async def async_update(self) -> None:
|
||||||
|
"""Refresh shade position."""
|
||||||
|
if self._update_in_progress:
|
||||||
|
# The update will likely timeout and
|
||||||
|
# error if are already have one in flight
|
||||||
|
return
|
||||||
|
await self._shade.refresh()
|
||||||
|
self._async_update_shade_data(self._shade.raw_data)
|
||||||
|
|
||||||
|
|
||||||
class PowerViewShade(PowerViewShadeBase):
|
class PowerViewShade(PowerViewShadeBase):
|
||||||
"""Represent a standard shade."""
|
"""Represent a standard shade."""
|
||||||
|
@ -10,6 +10,8 @@ from homeassistant.helpers.entity import DeviceInfo
|
|||||||
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
||||||
|
|
||||||
from .const import (
|
from .const import (
|
||||||
|
ATTR_BATTERY_KIND,
|
||||||
|
BATTERY_KIND_HARDWIRED,
|
||||||
DEVICE_FIRMWARE,
|
DEVICE_FIRMWARE,
|
||||||
DEVICE_MAC_ADDRESS,
|
DEVICE_MAC_ADDRESS,
|
||||||
DEVICE_MODEL,
|
DEVICE_MODEL,
|
||||||
@ -83,6 +85,9 @@ class ShadeEntity(HDEntity):
|
|||||||
super().__init__(coordinator, device_info, room_name, shade.id)
|
super().__init__(coordinator, device_info, room_name, shade.id)
|
||||||
self._shade_name = shade_name
|
self._shade_name = shade_name
|
||||||
self._shade = shade
|
self._shade = shade
|
||||||
|
self._is_hard_wired = bool(
|
||||||
|
shade.raw_data.get(ATTR_BATTERY_KIND) == BATTERY_KIND_HARDWIRED
|
||||||
|
)
|
||||||
|
|
||||||
@property
|
@property
|
||||||
def positions(self) -> PowerviewShadePositions:
|
def positions(self) -> PowerviewShadePositions:
|
||||||
@ -117,8 +122,3 @@ class ShadeEntity(HDEntity):
|
|||||||
device_info[ATTR_SW_VERSION] = sw_version
|
device_info[ATTR_SW_VERSION] = sw_version
|
||||||
|
|
||||||
return device_info
|
return device_info
|
||||||
|
|
||||||
async def async_update(self) -> None:
|
|
||||||
"""Refresh shade position."""
|
|
||||||
await self._shade.refresh()
|
|
||||||
self.data.update_shade_positions(self._shade.raw_data)
|
|
||||||
|
@ -91,3 +91,7 @@ class PowerViewShadeBatterySensor(ShadeEntity, SensorEntity):
|
|||||||
"""Update with new data from the coordinator."""
|
"""Update with new data from the coordinator."""
|
||||||
self._shade.raw_data = self.data.get_raw_data(self._shade.id)
|
self._shade.raw_data = self.data.get_raw_data(self._shade.id)
|
||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
|
|
||||||
|
async def async_update(self) -> None:
|
||||||
|
"""Refresh shade battery."""
|
||||||
|
await self._shade.refreshBattery()
|
||||||
|
Loading…
x
Reference in New Issue
Block a user