14 Commits

Author SHA1 Message Date
RyanClark123
ad91a851fc Update changelog and version
Update changelog and version
2024-09-26 10:18:03 +01:00
ryanbdclark
339dc43d6d Merge pull request #18 from Julien80/patch-1
Add translation fr
2024-09-26 10:14:03 +01:00
Julien80
f3c853e2d7 Add translation fr 2024-09-26 10:26:00 +02:00
ryanbdclark
dfc2ffc0e1 Update hacs.json 2024-07-09 07:29:05 +01:00
ryanbdclark
dc28ebb02f Update CHANGELOG.md 2024-06-17 09:31:30 +01:00
RyanClark123
52710ba7de Bumping pyowletapi
* Fix
Bump pyowletapi to 2024.6.1
2024-06-17 09:28:22 +01:00
ryanbdclark
d8449c14a4 Update CHANGELOG.md 2024-05-13 13:49:23 +01:00
ryanbdclark
3610262855 Update CHANGELOG.md 2024-05-13 13:48:02 +01:00
ryanbdclark
904023e93a Update CHANGELOG.md 2024-05-13 13:47:23 +01:00
RyanClark123
ceade24851 Sensors now show unavailable, refactoring
### Feature
* As per HA core patterns, certain sensors will now show as unavailable when sock is charging
###
* Refactoring as per core maintainers suggestions
2024-05-13 13:41:51 +01:00
ryanbdclark
ab486d5519 Merge pull request #14 from coreywillwhat/patch-1
fix: O2 Sat 10m Avg reporting `255%`
2024-05-13 09:52:30 +01:00
coreywillwhat
5e17ecdeb2 fix: O2 Sat 10m Avg reporting 255%
Just a suggestion!
The O2 Sat 10m average reports `255%` when not charging, and before it can calculate the 10m average. Suggest changing to report `None` until the average can be displayed. This way the graphs/data aren't skewed by the 255 value.
2024-05-07 11:32:15 -06:00
ryanbdclark
4b90ce0d61 Update CHANGELOG.md 2024-03-27 15:53:56 +00:00
RyanClark123
50c55dcfd3 Add binary sensor, correct interval input
### Feature
*Base station on added as binary sensor

### Fix
* Bumping pyowletapi to 2024.3.2
* UI config now allows you to set interval to 5 seconds, previously the minimum was 10
2024-03-27 15:51:12 +00:00
11 changed files with 301 additions and 64 deletions

View File

@@ -1,7 +1,31 @@
# Changelog # Changelog
<!--next-version-placeholder--> <!--next-version-placeholder-->
##2023.11.2 (2023-11-23) ## 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))
### Fix
* 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)
### Feature ### Feature
* Support added for V2 sock ([`50fe1a8`](https://github.com/ryanbdclark/owlet/commit/50fe1a87656b7d6413d06f06f3650fd0bfb48e02)) * 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)) * Added tests for binary sensors ([`50fe1a8`](https://github.com/ryanbdclark/owlet/commit/50fe1a87656b7d6413d06f06f3650fd0bfb48e02))

View File

@@ -1,4 +1,5 @@
"""The Owlet Smart Sock integration.""" """The Owlet Smart Sock integration."""
from __future__ import annotations from __future__ import annotations
import asyncio import asyncio
@@ -72,15 +73,12 @@ async def async_setup_entry(hass: HomeAssistant, entry: ConfigEntry) -> bool:
entry, data={**entry.data, **devices["tokens"]} 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) scan_interval = entry.options.get(CONF_SCAN_INTERVAL)
coordinators = { coordinators = {
serial: OwletCoordinator(hass, sock, scan_interval, entry) device["device"]["dsn"]: OwletCoordinator(
for (serial, sock) in socks.items() hass, Sock(owlet_api, device["device"]), scan_interval, entry
)
for device in devices["response"]
} }
await asyncio.gather( await asyncio.gather(

View File

@@ -1,4 +1,5 @@
"""Support for Owlet binary sensors.""" """Support for Owlet binary sensors."""
from __future__ import annotations from __future__ import annotations
from dataclasses import dataclass from dataclasses import dataclass
@@ -17,71 +18,85 @@ from .coordinator import OwletCoordinator
from .entity import OwletBaseEntity from .entity import OwletBaseEntity
@dataclass @dataclass(kw_only=True)
class OwletBinarySensorEntityDescription(BinarySensorEntityDescription): class OwletBinarySensorEntityDescription(BinarySensorEntityDescription):
"""Represent the owlet binary sensor entity description.""" """Represent the owlet binary sensor entity description."""
available_during_charging: bool
SENSORS: tuple[OwletBinarySensorEntityDescription, ...] = ( SENSORS: tuple[OwletBinarySensorEntityDescription, ...] = (
OwletBinarySensorEntityDescription( OwletBinarySensorEntityDescription(
key="charging", key="charging",
translation_key="charging", translation_key="charging",
device_class=BinarySensorDeviceClass.BATTERY_CHARGING, device_class=BinarySensorDeviceClass.BATTERY_CHARGING,
available_during_charging=True,
), ),
OwletBinarySensorEntityDescription( OwletBinarySensorEntityDescription(
key="high_heart_rate_alert", key="high_heart_rate_alert",
translation_key="high_hr_alrt", translation_key="high_hr_alrt",
device_class=BinarySensorDeviceClass.SOUND, device_class=BinarySensorDeviceClass.SOUND,
available_during_charging=True,
), ),
OwletBinarySensorEntityDescription( OwletBinarySensorEntityDescription(
key="low_heart_rate_alert", key="low_heart_rate_alert",
translation_key="low_hr_alrt", translation_key="low_hr_alrt",
device_class=BinarySensorDeviceClass.SOUND, device_class=BinarySensorDeviceClass.SOUND,
available_during_charging=True,
), ),
OwletBinarySensorEntityDescription( OwletBinarySensorEntityDescription(
key="high_oxygen_alert", key="high_oxygen_alert",
translation_key="high_ox_alrt", translation_key="high_ox_alrt",
device_class=BinarySensorDeviceClass.SOUND, device_class=BinarySensorDeviceClass.SOUND,
available_during_charging=True,
), ),
OwletBinarySensorEntityDescription( OwletBinarySensorEntityDescription(
key="low_oxygen_alert", key="low_oxygen_alert",
translation_key="low_ox_alrt", translation_key="low_ox_alrt",
device_class=BinarySensorDeviceClass.SOUND, device_class=BinarySensorDeviceClass.SOUND,
available_during_charging=True,
), ),
OwletBinarySensorEntityDescription( OwletBinarySensorEntityDescription(
key="critical_oxygen_alert", key="critical_oxygen_alert",
translation_key="crit_ox_alrt", translation_key="crit_ox_alrt",
device_class=BinarySensorDeviceClass.SOUND, device_class=BinarySensorDeviceClass.SOUND,
available_during_charging=True,
), ),
OwletBinarySensorEntityDescription( OwletBinarySensorEntityDescription(
key="low_battery_alert", key="low_battery_alert",
translation_key="low_batt_alrt", translation_key="low_batt_alrt",
device_class=BinarySensorDeviceClass.SOUND, device_class=BinarySensorDeviceClass.SOUND,
available_during_charging=True,
), ),
OwletBinarySensorEntityDescription( OwletBinarySensorEntityDescription(
key="critical_battery_alert", key="critical_battery_alert",
translation_key="crit_batt_alrt", translation_key="crit_batt_alrt",
device_class=BinarySensorDeviceClass.SOUND, device_class=BinarySensorDeviceClass.SOUND,
available_during_charging=True,
), ),
OwletBinarySensorEntityDescription( OwletBinarySensorEntityDescription(
key="lost_power_alert", key="lost_power_alert",
translation_key="lost_pwr_alrt", translation_key="lost_pwr_alrt",
device_class=BinarySensorDeviceClass.SOUND, device_class=BinarySensorDeviceClass.SOUND,
available_during_charging=True,
), ),
OwletBinarySensorEntityDescription( OwletBinarySensorEntityDescription(
key="sock_disconnected", key="sock_disconnected",
translation_key="sock_discon_alrt", translation_key="sock_discon_alrt",
device_class=BinarySensorDeviceClass.SOUND, device_class=BinarySensorDeviceClass.SOUND,
available_during_charging=True,
), ),
OwletBinarySensorEntityDescription( OwletBinarySensorEntityDescription(
key="sock_off", key="sock_off",
translation_key="sock_off", translation_key="sock_off",
device_class=BinarySensorDeviceClass.POWER, device_class=BinarySensorDeviceClass.POWER,
available_during_charging=True,
), ),
OwletBinarySensorEntityDescription( OwletBinarySensorEntityDescription(
key="sleep_state", key="base_station_on",
translation_key="awake", translation_key="base_on",
icon="mdi:sleep", device_class=BinarySensorDeviceClass.POWER,
available_during_charging=True,
), ),
) )
@@ -97,11 +112,13 @@ async def async_setup_entry(
sensors = [] sensors = []
for coordinator in coordinators: for coordinator in coordinators:
print(coordinator.sock.properties)
for sensor in SENSORS: for sensor in SENSORS:
if sensor.key in coordinator.sock.properties: if sensor.key in coordinator.sock.properties:
sensors.append(OwletBinarySensor(coordinator, sensor)) sensors.append(OwletBinarySensor(coordinator, sensor))
if OwletAwakeSensor.entity_description.key in coordinator.sock.properties:
sensors.append(OwletAwakeSensor(coordinator))
async_add_entities(sensors) async_add_entities(sensors)
@@ -118,11 +135,17 @@ class OwletBinarySensor(OwletBaseEntity, BinarySensorEntity):
self.entity_description = description self.entity_description = description
self._attr_unique_id = f"{self.sock.serial}-{description.key}" 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 @property
def is_on(self) -> bool: def is_on(self) -> bool:
"""Return true if the binary sensor is on.""" """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.entity_description.key == "sleep_state":
if self.sock.properties["charging"]: if self.sock.properties["charging"]:
return None return None
@@ -131,4 +154,27 @@ class OwletBinarySensor(OwletBaseEntity, BinarySensorEntity):
else: else:
state = True 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

View File

@@ -130,9 +130,9 @@ class OwletConfigFlow(config_entries.ConfigFlow, domain=DOMAIN):
self.reauth_entry, data={**entry_data, **token} self.reauth_entry, data={**entry_data, **token}
) )
await self.hass.config_entries.async_reload(self.reauth_entry.entry_id) await self.hass.config_entries.async_reload(self.reauth_entry.entry_id)
return self.async_abort(reason="reauth_successful") return self.async_abort(reason="reauth_successful")
except OwletPasswordError: except OwletPasswordError:
errors[CONF_PASSWORD] = "invalid_password" errors[CONF_PASSWORD] = "invalid_password"
@@ -165,7 +165,7 @@ class OptionsFlowHandler(config_entries.OptionsFlow):
vol.Required( vol.Required(
CONF_SCAN_INTERVAL, CONF_SCAN_INTERVAL,
default=self.config_entry.options.get(CONF_SCAN_INTERVAL), default=self.config_entry.options.get(CONF_SCAN_INTERVAL),
): vol.All(vol.Coerce(int), vol.Range(min=10)), ): vol.All(vol.Coerce(int), vol.Range(min=5)),
} }
) )

View File

@@ -9,7 +9,7 @@
"iot_class": "cloud_polling", "iot_class": "cloud_polling",
"issue_tracker": "https://github.com/ryanbdclark/owlet/issues", "issue_tracker": "https://github.com/ryanbdclark/owlet/issues",
"requirements": [ "requirements": [
"pyowletapi==2023.11.4" "pyowletapi==2024.6.1"
], ],
"version": "2023.11.2" "version": "2024.9.1"
} }

View File

@@ -1,4 +1,5 @@
"""Support for Owlet sensors.""" """Support for Owlet sensors."""
from __future__ import annotations from __future__ import annotations
from dataclasses import dataclass from dataclasses import dataclass
@@ -25,10 +26,12 @@ from .coordinator import OwletCoordinator
from .entity import OwletBaseEntity from .entity import OwletBaseEntity
@dataclass @dataclass(kw_only=True)
class OwletSensorEntityDescription(SensorEntityDescription): class OwletSensorEntityDescription(SensorEntityDescription):
"""Represent the owlet sensor entity description.""" """Represent the owlet sensor entity description."""
available_during_charging: bool
SENSORS: tuple[OwletSensorEntityDescription, ...] = ( SENSORS: tuple[OwletSensorEntityDescription, ...] = (
OwletSensorEntityDescription( OwletSensorEntityDescription(
@@ -37,6 +40,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
native_unit_of_measurement=PERCENTAGE, native_unit_of_measurement=PERCENTAGE,
device_class=SensorDeviceClass.BATTERY, device_class=SensorDeviceClass.BATTERY,
state_class=SensorStateClass.MEASUREMENT, state_class=SensorStateClass.MEASUREMENT,
available_during_charging=True,
), ),
OwletSensorEntityDescription( OwletSensorEntityDescription(
key="oxygen_saturation", key="oxygen_saturation",
@@ -44,6 +48,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
native_unit_of_measurement=PERCENTAGE, native_unit_of_measurement=PERCENTAGE,
state_class=SensorStateClass.MEASUREMENT, state_class=SensorStateClass.MEASUREMENT,
icon="mdi:leaf", icon="mdi:leaf",
available_during_charging=False,
), ),
OwletSensorEntityDescription( OwletSensorEntityDescription(
key="heart_rate", key="heart_rate",
@@ -51,13 +56,14 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
native_unit_of_measurement="bpm", native_unit_of_measurement="bpm",
state_class=SensorStateClass.MEASUREMENT, state_class=SensorStateClass.MEASUREMENT,
icon="mdi:heart-pulse", icon="mdi:heart-pulse",
available_during_charging=False,
), ),
OwletSensorEntityDescription( OwletSensorEntityDescription(
key="battery_minutes", key="battery_minutes",
translation_key="batterymin", translation_key="batterymin",
native_unit_of_measurement=UnitOfTime.MINUTES, native_unit_of_measurement=UnitOfTime.MINUTES,
device_class=SensorDeviceClass.DURATION, device_class=SensorDeviceClass.DURATION,
state_class=SensorStateClass.MEASUREMENT, available_during_charging=False,
), ),
OwletSensorEntityDescription( OwletSensorEntityDescription(
key="signal_strength", key="signal_strength",
@@ -65,6 +71,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
native_unit_of_measurement=SIGNAL_STRENGTH_DECIBELS_MILLIWATT, native_unit_of_measurement=SIGNAL_STRENGTH_DECIBELS_MILLIWATT,
device_class=SensorDeviceClass.SIGNAL_STRENGTH, device_class=SensorDeviceClass.SIGNAL_STRENGTH,
state_class=SensorStateClass.MEASUREMENT, state_class=SensorStateClass.MEASUREMENT,
available_during_charging=True,
), ),
OwletSensorEntityDescription( OwletSensorEntityDescription(
key="skin_temperature", key="skin_temperature",
@@ -72,11 +79,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
native_unit_of_measurement=UnitOfTemperature.CELSIUS, native_unit_of_measurement=UnitOfTemperature.CELSIUS,
device_class=SensorDeviceClass.TEMPERATURE, device_class=SensorDeviceClass.TEMPERATURE,
state_class=SensorStateClass.MEASUREMENT, state_class=SensorStateClass.MEASUREMENT,
), available_during_charging=False,
OwletSensorEntityDescription(
key="sleep_state",
translation_key="sleepstate",
device_class=SensorDeviceClass.ENUM,
), ),
OwletSensorEntityDescription( OwletSensorEntityDescription(
key="movement", key="movement",
@@ -84,13 +87,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
state_class=SensorStateClass.MEASUREMENT, state_class=SensorStateClass.MEASUREMENT,
icon="mdi:cursor-move", icon="mdi:cursor-move",
entity_registry_enabled_default=False, entity_registry_enabled_default=False,
), available_during_charging=False,
OwletSensorEntityDescription(
key="oxygen_10_av",
translation_key="o2saturation10a",
native_unit_of_measurement=PERCENTAGE,
state_class=SensorStateClass.MEASUREMENT,
icon="mdi:leaf",
), ),
OwletSensorEntityDescription( OwletSensorEntityDescription(
key="movement_bucket", key="movement_bucket",
@@ -98,6 +95,7 @@ SENSORS: tuple[OwletSensorEntityDescription, ...] = (
state_class=SensorStateClass.MEASUREMENT, state_class=SensorStateClass.MEASUREMENT,
icon="mdi:bucket-outline", icon="mdi:bucket-outline",
entity_registry_enabled_default=False, entity_registry_enabled_default=False,
available_during_charging=False,
), ),
) )
@@ -120,6 +118,11 @@ async def async_setup_entry(
if sensor.key in coordinator.sock.properties: if sensor.key in coordinator.sock.properties:
sensors.append(OwletSensor(coordinator, sensor)) 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) async_add_entities(sensors)
@@ -136,32 +139,68 @@ class OwletSensor(OwletBaseEntity, SensorEntity):
self.entity_description: OwletSensorEntityDescription = description self.entity_description: OwletSensorEntityDescription = description
self._attr_unique_id = f"{self.sock.serial}-{description.key}" 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 @property
def native_value(self) -> StateType: def native_value(self) -> StateType:
"""Return sensor value.""" """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] 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 @property
def options(self) -> list[str] | None: def native_value(self) -> StateType:
"""Set options for sleep state.""" """Return sensor value."""
if self.entity_description.key != "sleep_state": return SLEEP_STATES[self.sock.properties["sleep_state"]]
return None
return list(SLEEP_STATES.values())
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,
)
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)

View File

@@ -5,13 +5,13 @@
"data": { "data": {
"region": "Region", "region": "Region",
"username": "Email", "username": "Email",
"password": "Password" "password": "[%key:common::config_flow::data::password%]"
} }
}, },
"reauth_confirm": { "reauth_confirm": {
"title": "Reauthentiaction required for Owlet", "title": "Reauthentiaction required for Owlet",
"data": { "data": {
"password": "Password" "password": "[%key:common::config_flow::data::password%]"
} }
} }
}, },
@@ -74,6 +74,9 @@
}, },
"awake": { "awake": {
"name": "Awake" "name": "Awake"
},
"base_on": {
"name": "Base station on"
} }
}, },
"sensor": { "sensor": {

View File

@@ -5,13 +5,13 @@
"data": { "data": {
"region": "Region", "region": "Region",
"username": "Email", "username": "Email",
"password": "Password" "password": "[%key:common::config_flow::data::password%]"
} }
}, },
"reauth_confirm": { "reauth_confirm": {
"title": "Reauthentiaction required for Owlet", "title": "Reauthentiaction required for Owlet",
"data": { "data": {
"password": "Password" "password": "[%key:common::config_flow::data::password%]"
} }
} }
}, },
@@ -74,6 +74,9 @@
}, },
"awake": { "awake": {
"name": "Awake" "name": "Awake"
},
"base_on": {
"name": "Base station on"
} }
}, },
"sensor": { "sensor": {

View File

@@ -0,0 +1,121 @@
{
"config": {
"step": {
"user": {
"data": {
"region": "Région",
"username": "Email",
"password": "[%key:common::config_flow::data::password%]"
}
},
"reauth_confirm": {
"title": "Réauthentification requise pour Owlet",
"data": {
"password": "[%key:common::config_flow::data::password%]"
}
}
},
"error": {
"cannot_connect": "[%key:common::config_flow::error::cannot_connect%]",
"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é"
},
"base_on": {
"name": "Station de base allumée"
}
},
"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"
}
}
}
}

View File

@@ -5,13 +5,13 @@
"data": { "data": {
"region": "Region", "region": "Region",
"username": "Email", "username": "Email",
"password": "Password" "password": "[%key:common::config_flow::data::password%]"
} }
}, },
"reauth_confirm": { "reauth_confirm": {
"title": "Reauthentiaction required for Owlet", "title": "Reauthentiaction required for Owlet",
"data": { "data": {
"password": "Password" "password": "[%key:common::config_flow::data::password%]"
} }
} }
}, },
@@ -74,6 +74,9 @@
}, },
"awake": { "awake": {
"name": "Awake" "name": "Awake"
},
"base_on": {
"name": "Base station on"
} }
}, },
"sensor": { "sensor": {

View File

@@ -1,7 +1,7 @@
{ {
"name": "Owlet", "name": "Owlet",
"hacs": "1.32.1", "hacs": "1.32.1",
"homeassistant": "2023.04.1", "homeassistant": "2024.1",
"zip_release": true, "zip_release": true,
"filename": "owlet.zip" "filename": "owlet.zip"
} }