Compare commits
17 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
e28b9ddf3e | ||
|
|
f63e0a6dfe | ||
|
|
82823be1c8 | ||
|
|
14787e03c4 | ||
|
|
0991eb31d9 | ||
|
|
ad91a851fc | ||
|
|
339dc43d6d | ||
|
|
f3c853e2d7 | ||
|
|
dfc2ffc0e1 | ||
|
|
dc28ebb02f | ||
|
|
52710ba7de | ||
|
|
d8449c14a4 | ||
|
|
3610262855 | ||
|
|
904023e93a | ||
|
|
ceade24851 | ||
|
|
ab486d5519 | ||
|
|
5e17ecdeb2 |
28
CHANGELOG.md
28
CHANGELOG.md
@@ -1,7 +1,31 @@
|
||||
# Changelog
|
||||
|
||||
<!--next-version-placeholder-->
|
||||
##2024.3.1
|
||||
## 2024.10.1 (2024-10-09)
|
||||
### Feature
|
||||
* Base station has now been removed from binary sensors and added as a switch ([`f63e0a6`](https://github.com/ryanbdclark/owlet/commit/f63e0a6dfeab1a05ba09ef3e0087cb404ba0dac4))
|
||||
### Fix
|
||||
* Bump pyowletapi to 2024.10.1 ([`82823be`](https://github.com/ryanbdclark/owlet/commit/82823be1c8265d2b9431771136853febef648650))
|
||||
* Fix strings for password and connection error in all languages ([`14787e0`](https://github.com/ryanbdclark/owlet/commit/14787e03c4d275f46f446921a3ee133fc7cfd1b1))
|
||||
* Add state class to battery minutes and O2 saturation 10 minute average ([`0991eb3`](https://github.com/ryanbdclark/owlet/commit/0991eb31d919f3ee9f65ece793166d7ee3e33c38))
|
||||
|
||||
## 2024.9.1 (2024-09-26)
|
||||
### Feature
|
||||
* Now includes French translation, thanks [`@Julien80`](https://github.com/Julien80) ([`f3c853e`](https://github.com/ryanbdclark/owlet/commit/f3c853e2d7243d766889f2d18c718819da30e4be))
|
||||
|
||||
## 2024.6.1 (2024-06-17)
|
||||
### Fix
|
||||
* Bumping pyowletapi to 2024.6.1 to resolve setup errors ([`52710ba`](https://github.com/ryanbdclark/owlet/commit/52710ba7de53fd07195537c2a5fd2f95bc7dfd1a))
|
||||
|
||||
## 2024.5.1 (2024-05-13)
|
||||
### Feature
|
||||
* As per HA core patterns, certain sensors will now show as unavailable when sock is charging ([`ceade24`](https://github.com/ryanbdclark/owlet/commit/ceade24851479b8c9bc60b7b8bed74a7bdb927e9))
|
||||
* Oxygen 10 minute average now only shows a figure if it is between 0 and 100 this avoids skewing by 255 values before the 10 minutes is reached, thanks @coreywillwhat ([`5e17ecd`](https://github.com/ryanbdclark/owlet/commit/5e17ecdeb2aca5bbb35f19ca5795a2c5e0f776ab))
|
||||
### Fix
|
||||
* Refactoring as per core maintainers suggestions ([`ceade24`](https://github.com/ryanbdclark/owlet/commit/ceade24851479b8c9bc60b7b8bed74a7bdb927e9))
|
||||
|
||||
|
||||
## 2024.3.1
|
||||
### Feature
|
||||
* Base station on added as binary sensor ([`50c55dc`](https://github.com/ryanbdclark/owlet/commit/50c55dcfd30d15027155a8f1d05340238501522d))
|
||||
|
||||
@@ -9,7 +33,7 @@
|
||||
* Bumping pyowletapi to 2024.3.2 ([`50c55dc`](https://github.com/ryanbdclark/owlet/commit/50c55dcfd30d15027155a8f1d05340238501522d))
|
||||
* UI config now allows you to set interval to 5 seconds, previously the minimum was 10 ([`50c55dc`](https://github.com/ryanbdclark/owlet/commit/50c55dcfd30d15027155a8f1d05340238501522d))
|
||||
|
||||
##2023.11.2 (2023-11-23)
|
||||
## 2023.11.2 (2023-11-23)
|
||||
### Feature
|
||||
* Support added for V2 sock ([`50fe1a8`](https://github.com/ryanbdclark/owlet/commit/50fe1a87656b7d6413d06f06f3650fd0bfb48e02))
|
||||
* Added tests for binary sensors ([`50fe1a8`](https://github.com/ryanbdclark/owlet/commit/50fe1a87656b7d6413d06f06f3650fd0bfb48e02))
|
||||
|
||||
@@ -1,4 +1,5 @@
|
||||
"""The Owlet Smart Sock integration."""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
import asyncio
|
||||
@@ -29,7 +30,7 @@ from homeassistant.helpers.aiohttp_client import async_get_clientsession
|
||||
from .const import CONF_OWLET_EXPIRY, CONF_OWLET_REFRESH, DOMAIN, SUPPORTED_VERSIONS
|
||||
from .coordinator import OwletCoordinator
|
||||
|
||||
PLATFORMS: list[Platform] = [Platform.BINARY_SENSOR, Platform.SENSOR]
|
||||
PLATFORMS: list[Platform] = [Platform.BINARY_SENSOR, Platform.SENSOR, Platform.SWITCH]
|
||||
|
||||
_LOGGER = logging.getLogger(__name__)
|
||||
|
||||
@@ -72,15 +73,12 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
|
||||
entry, data={**entry.data, **devices["tokens"]}
|
||||
)
|
||||
|
||||
socks = {
|
||||
device["device"]["dsn"]: Sock(owlet_api, device["device"])
|
||||
for device in devices["response"]
|
||||
}
|
||||
|
||||
scan_interval = entry.options.get(CONF_SCAN_INTERVAL)
|
||||
coordinators = {
|
||||
serial: OwletCoordinator(hass, sock, scan_interval, entry)
|
||||
for (serial, sock) in socks.items()
|
||||
device["device"]["dsn"]: OwletCoordinator(
|
||||
hass, Sock(owlet_api, device["device"]), scan_interval, entry
|
||||
)
|
||||
for device in devices["response"]
|
||||
}
|
||||
|
||||
await asyncio.gather(
|
||||
|
||||
@@ -1,4 +1,5 @@
|
||||
"""Support for Owlet binary sensors."""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
from dataclasses import dataclass
|
||||
@@ -17,76 +18,79 @@ from .coordinator import OwletCoordinator
|
||||
from .entity import OwletBaseEntity
|
||||
|
||||
|
||||
@dataclass
|
||||
@dataclass(kw_only=True)
|
||||
class OwletBinarySensorEntityDescription(BinarySensorEntityDescription):
|
||||
"""Represent the owlet binary sensor entity description."""
|
||||
|
||||
available_during_charging: bool
|
||||
|
||||
|
||||
SENSORS: tuple[OwletBinarySensorEntityDescription, ...] = (
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="charging",
|
||||
translation_key="charging",
|
||||
device_class=BinarySensorDeviceClass.BATTERY_CHARGING,
|
||||
available_during_charging=True,
|
||||
),
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="high_heart_rate_alert",
|
||||
translation_key="high_hr_alrt",
|
||||
device_class=BinarySensorDeviceClass.SOUND,
|
||||
available_during_charging=True,
|
||||
),
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="low_heart_rate_alert",
|
||||
translation_key="low_hr_alrt",
|
||||
device_class=BinarySensorDeviceClass.SOUND,
|
||||
available_during_charging=True,
|
||||
),
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="high_oxygen_alert",
|
||||
translation_key="high_ox_alrt",
|
||||
device_class=BinarySensorDeviceClass.SOUND,
|
||||
available_during_charging=True,
|
||||
),
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="low_oxygen_alert",
|
||||
translation_key="low_ox_alrt",
|
||||
device_class=BinarySensorDeviceClass.SOUND,
|
||||
available_during_charging=True,
|
||||
),
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="critical_oxygen_alert",
|
||||
translation_key="crit_ox_alrt",
|
||||
device_class=BinarySensorDeviceClass.SOUND,
|
||||
available_during_charging=True,
|
||||
),
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="low_battery_alert",
|
||||
translation_key="low_batt_alrt",
|
||||
device_class=BinarySensorDeviceClass.SOUND,
|
||||
available_during_charging=True,
|
||||
),
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="critical_battery_alert",
|
||||
translation_key="crit_batt_alrt",
|
||||
device_class=BinarySensorDeviceClass.SOUND,
|
||||
available_during_charging=True,
|
||||
),
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="lost_power_alert",
|
||||
translation_key="lost_pwr_alrt",
|
||||
device_class=BinarySensorDeviceClass.SOUND,
|
||||
available_during_charging=True,
|
||||
),
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="sock_disconnected",
|
||||
translation_key="sock_discon_alrt",
|
||||
device_class=BinarySensorDeviceClass.SOUND,
|
||||
available_during_charging=True,
|
||||
),
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="sock_off",
|
||||
translation_key="sock_off",
|
||||
device_class=BinarySensorDeviceClass.POWER,
|
||||
),
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="sleep_state",
|
||||
translation_key="awake",
|
||||
icon="mdi:sleep",
|
||||
),
|
||||
OwletBinarySensorEntityDescription(
|
||||
key="base_station_on",
|
||||
translation_key="base_on",
|
||||
device_class=BinarySensorDeviceClass.POWER,
|
||||
available_during_charging=True,
|
||||
),
|
||||
)
|
||||
|
||||
@@ -106,6 +110,9 @@ async def async_setup_entry(
|
||||
if sensor.key in coordinator.sock.properties:
|
||||
sensors.append(OwletBinarySensor(coordinator, sensor))
|
||||
|
||||
if OwletAwakeSensor.entity_description.key in coordinator.sock.properties:
|
||||
sensors.append(OwletAwakeSensor(coordinator))
|
||||
|
||||
async_add_entities(sensors)
|
||||
|
||||
|
||||
@@ -122,11 +129,17 @@ class OwletBinarySensor(OwletBaseEntity, BinarySensorEntity):
|
||||
self.entity_description = description
|
||||
self._attr_unique_id = f"{self.sock.serial}-{description.key}"
|
||||
|
||||
@property
|
||||
def available(self) -> bool:
|
||||
"""Return if entity is available."""
|
||||
return super().available and (
|
||||
not self.sock.properties["charging"]
|
||||
or self.entity_description.available_during_charging
|
||||
)
|
||||
|
||||
@property
|
||||
def is_on(self) -> bool:
|
||||
"""Return true if the binary sensor is on."""
|
||||
state = self.sock.properties[self.entity_description.key]
|
||||
|
||||
if self.entity_description.key == "sleep_state":
|
||||
if self.sock.properties["charging"]:
|
||||
return None
|
||||
@@ -135,4 +148,31 @@ class OwletBinarySensor(OwletBaseEntity, BinarySensorEntity):
|
||||
else:
|
||||
state = True
|
||||
|
||||
return state
|
||||
return self.sock.properties[self.entity_description.key]
|
||||
|
||||
|
||||
class OwletAwakeSensor(OwletBinarySensor):
|
||||
"""Representation of an Owlet sleep sensor."""
|
||||
|
||||
entity_description = OwletBinarySensorEntityDescription(
|
||||
key="sleep_state",
|
||||
translation_key="awake",
|
||||
icon="mdi:sleep",
|
||||
available_during_charging=False,
|
||||
)
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
coordinator: OwletCoordinator,
|
||||
) -> None:
|
||||
"""Initialize the sensor."""
|
||||
super().__init__(coordinator, self.entity_description)
|
||||
|
||||
@property
|
||||
def is_on(self) -> bool:
|
||||
"""Return true if the binary sensor is on."""
|
||||
return (
|
||||
False
|
||||
if self.sock.properties[self.entity_description.key] in [8, 15]
|
||||
else True
|
||||
)
|
||||
|
||||
@@ -9,7 +9,7 @@
|
||||
"iot_class": "cloud_polling",
|
||||
"issue_tracker": "https://github.com/ryanbdclark/owlet/issues",
|
||||
"requirements": [
|
||||
"pyowletapi==2024.3.2"
|
||||
"pyowletapi==2024.10.1"
|
||||
],
|
||||
"version": "2024.3.1"
|
||||
"version": "2024.10.1"
|
||||
}
|
||||
@@ -1,4 +1,5 @@
|
||||
"""Support for Owlet sensors."""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
from dataclasses import dataclass
|
||||
@@ -25,10 +26,12 @@ from .coordinator import OwletCoordinator
|
||||
from .entity import OwletBaseEntity
|
||||
|
||||
|
||||
@dataclass
|
||||
@dataclass(kw_only=True)
|
||||
class OwletSensorEntityDescription(SensorEntityDescription):
|
||||
"""Represent the owlet sensor entity description."""
|
||||
|
||||
available_during_charging: bool
|
||||
|
||||
|
||||
SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
||||
OwletSensorEntityDescription(
|
||||
@@ -37,6 +40,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
||||
native_unit_of_measurement=PERCENTAGE,
|
||||
device_class=SensorDeviceClass.BATTERY,
|
||||
state_class=SensorStateClass.MEASUREMENT,
|
||||
available_during_charging=True,
|
||||
),
|
||||
OwletSensorEntityDescription(
|
||||
key="oxygen_saturation",
|
||||
@@ -44,6 +48,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
||||
native_unit_of_measurement=PERCENTAGE,
|
||||
state_class=SensorStateClass.MEASUREMENT,
|
||||
icon="mdi:leaf",
|
||||
available_during_charging=False,
|
||||
),
|
||||
OwletSensorEntityDescription(
|
||||
key="heart_rate",
|
||||
@@ -51,6 +56,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
||||
native_unit_of_measurement="bpm",
|
||||
state_class=SensorStateClass.MEASUREMENT,
|
||||
icon="mdi:heart-pulse",
|
||||
available_during_charging=False,
|
||||
),
|
||||
OwletSensorEntityDescription(
|
||||
key="battery_minutes",
|
||||
@@ -58,6 +64,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
||||
native_unit_of_measurement=UnitOfTime.MINUTES,
|
||||
device_class=SensorDeviceClass.DURATION,
|
||||
state_class=SensorStateClass.MEASUREMENT,
|
||||
available_during_charging=False,
|
||||
),
|
||||
OwletSensorEntityDescription(
|
||||
key="signal_strength",
|
||||
@@ -65,6 +72,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
||||
native_unit_of_measurement=SIGNAL_STRENGTH_DECIBELS_MILLIWATT,
|
||||
device_class=SensorDeviceClass.SIGNAL_STRENGTH,
|
||||
state_class=SensorStateClass.MEASUREMENT,
|
||||
available_during_charging=True,
|
||||
),
|
||||
OwletSensorEntityDescription(
|
||||
key="skin_temperature",
|
||||
@@ -72,11 +80,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
||||
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
|
||||
device_class=SensorDeviceClass.TEMPERATURE,
|
||||
state_class=SensorStateClass.MEASUREMENT,
|
||||
),
|
||||
OwletSensorEntityDescription(
|
||||
key="sleep_state",
|
||||
translation_key="sleepstate",
|
||||
device_class=SensorDeviceClass.ENUM,
|
||||
available_during_charging=False,
|
||||
),
|
||||
OwletSensorEntityDescription(
|
||||
key="movement",
|
||||
@@ -84,13 +88,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
||||
state_class=SensorStateClass.MEASUREMENT,
|
||||
icon="mdi:cursor-move",
|
||||
entity_registry_enabled_default=False,
|
||||
),
|
||||
OwletSensorEntityDescription(
|
||||
key="oxygen_10_av",
|
||||
translation_key="o2saturation10a",
|
||||
native_unit_of_measurement=PERCENTAGE,
|
||||
state_class=SensorStateClass.MEASUREMENT,
|
||||
icon="mdi:leaf",
|
||||
available_during_charging=False,
|
||||
),
|
||||
OwletSensorEntityDescription(
|
||||
key="movement_bucket",
|
||||
@@ -98,6 +96,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
|
||||
state_class=SensorStateClass.MEASUREMENT,
|
||||
icon="mdi:bucket-outline",
|
||||
entity_registry_enabled_default=False,
|
||||
available_during_charging=False,
|
||||
),
|
||||
)
|
||||
|
||||
@@ -120,6 +119,14 @@ async def async_setup_entry(
|
||||
if sensor.key in coordinator.sock.properties:
|
||||
sensors.append(OwletSensor(coordinator, sensor))
|
||||
|
||||
if OwletSleepSensor.entity_description.key in coordinator.sock.properties:
|
||||
sensors.append(OwletSleepSensor(coordinator))
|
||||
if (
|
||||
OwletOxygenAverageSensor.entity_description.key
|
||||
in coordinator.sock.properties
|
||||
):
|
||||
sensors.append(OwletOxygenAverageSensor(coordinator))
|
||||
|
||||
async_add_entities(sensors)
|
||||
|
||||
|
||||
@@ -136,32 +143,75 @@ class OwletSensor(OwletBaseEntity, SensorEntity):
|
||||
self.entity_description: OwletSensorEntityDescription = description
|
||||
self._attr_unique_id = f"{self.sock.serial}-{description.key}"
|
||||
|
||||
@property
|
||||
def available(self) -> bool:
|
||||
"""Return if entity is available."""
|
||||
return super().available and (
|
||||
not self.sock.properties["charging"]
|
||||
or self.entity_description.available_during_charging
|
||||
)
|
||||
|
||||
@property
|
||||
def native_value(self) -> StateType:
|
||||
"""Return sensor value."""
|
||||
|
||||
if (
|
||||
self.entity_description.key
|
||||
in [
|
||||
"heart_rate",
|
||||
"battery_minutes",
|
||||
"oxygen_saturation",
|
||||
"skin_temperature",
|
||||
"oxygen_10_av",
|
||||
"sleep_state",
|
||||
]
|
||||
and self.sock.properties["charging"]
|
||||
):
|
||||
return None
|
||||
|
||||
if self.entity_description.key == "sleep_state":
|
||||
return SLEEP_STATES[self.sock.properties["sleep_state"]]
|
||||
|
||||
return self.sock.properties[self.entity_description.key]
|
||||
|
||||
|
||||
class OwletSleepSensor(OwletSensor):
|
||||
"""Representation of an Owlet sleep sensor."""
|
||||
|
||||
_attr_options = list(SLEEP_STATES.values())
|
||||
entity_description = OwletSensorEntityDescription(
|
||||
key="sleep_state",
|
||||
translation_key="sleepstate",
|
||||
device_class=SensorDeviceClass.ENUM,
|
||||
available_during_charging=False,
|
||||
)
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
coordinator: OwletCoordinator,
|
||||
) -> None:
|
||||
"""Initialize the sensor."""
|
||||
super().__init__(coordinator, self.entity_description)
|
||||
|
||||
@property
|
||||
def options(self) -> list[str] | None:
|
||||
"""Set options for sleep state."""
|
||||
if self.entity_description.key != "sleep_state":
|
||||
return None
|
||||
return list(SLEEP_STATES.values())
|
||||
def native_value(self) -> StateType:
|
||||
"""Return sensor value."""
|
||||
return SLEEP_STATES[self.sock.properties["sleep_state"]]
|
||||
|
||||
|
||||
class OwletOxygenAverageSensor(OwletSensor):
|
||||
"""Representation of an Owlet sleep sensor."""
|
||||
|
||||
entity_description = OwletSensorEntityDescription(
|
||||
key="oxygen_10_av",
|
||||
translation_key="o2saturation10a",
|
||||
native_unit_of_measurement=PERCENTAGE,
|
||||
icon="mdi:leaf",
|
||||
available_during_charging=False,
|
||||
state_class=SensorStateClass.MEASUREMENT,
|
||||
)
|
||||
|
||||
def __init__(
|
||||
self,
|
||||
coordinator: OwletCoordinator,
|
||||
) -> None:
|
||||
"""Initialize the sensor."""
|
||||
super().__init__(coordinator, self.entity_description)
|
||||
|
||||
@property
|
||||
def available(self) -> bool:
|
||||
"""Return if entity is available."""
|
||||
return (
|
||||
super().available
|
||||
and (
|
||||
not self.sock.properties["charging"]
|
||||
or self.entity_description.available_during_charging
|
||||
)
|
||||
and (
|
||||
self.sock.properties["oxygen_10_av"] >= 0
|
||||
and self.sock.properties["oxygen_10_av"] <= 100
|
||||
)
|
||||
)
|
||||
|
||||
@@ -5,13 +5,13 @@
|
||||
"data": {
|
||||
"region": "Region",
|
||||
"username": "Email",
|
||||
"password": "Password"
|
||||
"password": "[%key:common::config_flow::data::password%]"
|
||||
}
|
||||
},
|
||||
"reauth_confirm": {
|
||||
"title": "Reauthentiaction required for Owlet",
|
||||
"data": {
|
||||
"password": "Password"
|
||||
"password": "[%key:common::config_flow::data::password%]"
|
||||
}
|
||||
}
|
||||
},
|
||||
@@ -74,9 +74,6 @@
|
||||
},
|
||||
"awake": {
|
||||
"name": "Awake"
|
||||
},
|
||||
"base_on": {
|
||||
"name": "Base station on"
|
||||
}
|
||||
},
|
||||
"sensor": {
|
||||
@@ -116,6 +113,11 @@
|
||||
"movementbucket": {
|
||||
"name": "Movement bucket"
|
||||
}
|
||||
},
|
||||
"switch": {
|
||||
"base_on": {
|
||||
"name": "Base station on"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
58
custom_components/owlet/switch.py
Normal file
58
custom_components/owlet/switch.py
Normal file
@@ -0,0 +1,58 @@
|
||||
"""Support for Owlet switches."""
|
||||
|
||||
from __future__ import annotations
|
||||
|
||||
from dataclasses import dataclass
|
||||
from datetime import timedelta
|
||||
from typing import Any
|
||||
|
||||
from homeassistant.config_entries import ConfigEntry
|
||||
from homeassistant.components.switch import SwitchEntity, SwitchEntityDescription
|
||||
from homeassistant.core import HomeAssistant
|
||||
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
||||
|
||||
from .const import DOMAIN
|
||||
from .coordinator import OwletCoordinator
|
||||
from .entity import OwletBaseEntity
|
||||
|
||||
SCAN_INTERVAL = timedelta(seconds=5)
|
||||
PARALLEL_UPDATES = 0
|
||||
|
||||
|
||||
async def async_setup_entry(
|
||||
hass: HomeAssistant,
|
||||
config_entry: ConfigEntry,
|
||||
async_add_entities: AddEntitiesCallback,
|
||||
) -> None:
|
||||
"""Set up Owlet switch based on a config entry."""
|
||||
coordinators: OwletCoordinator = hass.data[DOMAIN][config_entry.entry_id].values()
|
||||
|
||||
switches = []
|
||||
for coordinator in coordinators:
|
||||
switches.append(OwletBaseSwitch(coordinator))
|
||||
async_add_entities(switches)
|
||||
|
||||
|
||||
class OwletBaseSwitch(OwletBaseEntity, SwitchEntity):
|
||||
"""Defines a Owlet switch."""
|
||||
|
||||
_attr_has_entity_name = True
|
||||
_attr_translation_key = "base_on"
|
||||
|
||||
def __init__(self, coordinator: OwletCoordinator) -> None:
|
||||
"""Initialize ecobee ventilator platform."""
|
||||
super().__init__(coordinator)
|
||||
self._attr_unique_id = f"{self.sock.serial}-base_station_on"
|
||||
self._attr_is_on = False
|
||||
|
||||
@property
|
||||
def is_on(self):
|
||||
return self.sock.properties["base_station_on"]
|
||||
|
||||
async def async_turn_on(self, **kwargs: Any) -> None:
|
||||
"""Turn on the switch."""
|
||||
await self.sock.control_base_station(True)
|
||||
|
||||
async def async_turn_off(self, **kwargs: Any) -> None:
|
||||
"""Turn off the switch."""
|
||||
await self.sock.control_base_station(False)
|
||||
@@ -16,7 +16,7 @@
|
||||
}
|
||||
},
|
||||
"error": {
|
||||
"cannot_connect": "[%key:common::config_flow::error::cannot_connect%]",
|
||||
"cannot_connect": "Failed to connect",
|
||||
"invalid_email": "Entered email address is incorrect",
|
||||
"invalid_password": "Entered password is incorrect",
|
||||
"invalid_credentials": "Entered credentials are incorrect",
|
||||
@@ -74,9 +74,6 @@
|
||||
},
|
||||
"awake": {
|
||||
"name": "Awake"
|
||||
},
|
||||
"base_on": {
|
||||
"name": "Base station on"
|
||||
}
|
||||
},
|
||||
"sensor": {
|
||||
@@ -116,6 +113,11 @@
|
||||
"movementbucket": {
|
||||
"name": "Movement bucket"
|
||||
}
|
||||
},
|
||||
"switch": {
|
||||
"base_on": {
|
||||
"name": "Base station on"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
123
custom_components/owlet/translations/fr.json
Normal file
123
custom_components/owlet/translations/fr.json
Normal file
@@ -0,0 +1,123 @@
|
||||
{
|
||||
"config": {
|
||||
"step": {
|
||||
"user": {
|
||||
"data": {
|
||||
"region": "Région",
|
||||
"username": "Email",
|
||||
"password": "Mot de passe"
|
||||
}
|
||||
},
|
||||
"reauth_confirm": {
|
||||
"title": "Réauthentification requise pour Owlet",
|
||||
"data": {
|
||||
"password": "Mot de passe"
|
||||
}
|
||||
}
|
||||
},
|
||||
"error": {
|
||||
"cannot_connect": "N'a pas réussi à se connecter",
|
||||
"invalid_email": "L'adresse e-mail saisie est incorrecte",
|
||||
"invalid_password": "Le mot de passe saisi est incorrect",
|
||||
"invalid_credentials": "Les informations d'identification saisies sont incorrectes",
|
||||
"unknown": "Une erreur inconnue est survenue"
|
||||
},
|
||||
"abort": {
|
||||
"already_configured": "Appareil déjà configuré",
|
||||
"reauth_successful": "Réauthentification réussie"
|
||||
}
|
||||
},
|
||||
"options": {
|
||||
"step": {
|
||||
"init": {
|
||||
"title": "Configurer les options pour Owlet",
|
||||
"data": {
|
||||
"pollinterval": "Intervalle de sondage en secondes, minimum 10"
|
||||
}
|
||||
}
|
||||
}
|
||||
},
|
||||
"entity": {
|
||||
"binary_sensor": {
|
||||
"charging": {
|
||||
"name": "En charge"
|
||||
},
|
||||
"high_hr_alrt": {
|
||||
"name": "Alerte fréquence cardiaque élevée"
|
||||
},
|
||||
"low_hr_alrt": {
|
||||
"name": "Alerte fréquence cardiaque basse"
|
||||
},
|
||||
"high_ox_alrt": {
|
||||
"name": "Alerte oxygène élevé"
|
||||
},
|
||||
"low_ox_alrt": {
|
||||
"name": "Alerte oxygène faible"
|
||||
},
|
||||
"crit_ox_alrt": {
|
||||
"name": "Alerte oxygène critique"
|
||||
},
|
||||
"low_batt_alrt": {
|
||||
"name": "Alerte batterie faible"
|
||||
},
|
||||
"crit_batt_alrt": {
|
||||
"name": "Alerte batterie critique"
|
||||
},
|
||||
"lost_pwr_alrt": {
|
||||
"name": "Alerte perte d'alimentation"
|
||||
},
|
||||
"sock_discon_alrt": {
|
||||
"name": "Alerte chaussette déconnectée"
|
||||
},
|
||||
"sock_off": {
|
||||
"name": "Chaussette retirée"
|
||||
},
|
||||
"awake": {
|
||||
"name": "Réveillé"
|
||||
}
|
||||
},
|
||||
"sensor": {
|
||||
"batterypercent": {
|
||||
"name": "Pourcentage de batterie"
|
||||
},
|
||||
"signalstrength": {
|
||||
"name": "Force du signal"
|
||||
},
|
||||
"o2saturation": {
|
||||
"name": "Saturation O2"
|
||||
},
|
||||
"o2saturation10a": {
|
||||
"name": "Moyenne de saturation O2 sur 10 minutes"
|
||||
},
|
||||
"heartrate": {
|
||||
"name": "Fréquence cardiaque"
|
||||
},
|
||||
"batterymin": {
|
||||
"name": "Autonomie de la batterie restante"
|
||||
},
|
||||
"skintemp": {
|
||||
"name": "Température de la peau"
|
||||
},
|
||||
"sleepstate": {
|
||||
"name": "État de sommeil",
|
||||
"state": {
|
||||
"unknown": "Inconnu",
|
||||
"awake": "Réveillé",
|
||||
"light_sleep": "Sommeil léger",
|
||||
"deep_sleep": "Sommeil profond"
|
||||
}
|
||||
},
|
||||
"movement": {
|
||||
"name": "Mouvement"
|
||||
},
|
||||
"movementbucket": {
|
||||
"name": "Seuil de mouvement"
|
||||
}
|
||||
},
|
||||
"switch": {
|
||||
"base_on": {
|
||||
"name": "Station de base allumée"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
@@ -16,7 +16,7 @@
|
||||
}
|
||||
},
|
||||
"error": {
|
||||
"cannot_connect": "[%key:common::config_flow::error::cannot_connect%]",
|
||||
"cannot_connect": "Failed to connect",
|
||||
"invalid_email": "Entered email address is incorrect",
|
||||
"invalid_password": "Entered password is incorrect",
|
||||
"invalid_credentials": "Entered credentials are incorrect",
|
||||
@@ -74,9 +74,6 @@
|
||||
},
|
||||
"awake": {
|
||||
"name": "Awake"
|
||||
},
|
||||
"base_on": {
|
||||
"name": "Base station on"
|
||||
}
|
||||
},
|
||||
"sensor": {
|
||||
@@ -116,6 +113,11 @@
|
||||
"movementbucket": {
|
||||
"name": "Movement bucket"
|
||||
}
|
||||
},
|
||||
"switch": {
|
||||
"base_on": {
|
||||
"name": "Base station on"
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user