Compare commits
16 Commits
v0.3.0
...
v0.5.0-bet
Author | SHA1 | Date | |
---|---|---|---|
4901be4050 | |||
c8189414b8 | |||
799ac67d94 | |||
7e9202ef38 | |||
e1a2af70e9 | |||
c78aeb1fbb | |||
67f280512d | |||
6093b57f76 | |||
287e895f4d | |||
d3dc1b9f45 | |||
116f9d5470 | |||
05e9d835b1 | |||
135d6cafed | |||
077bded6dd | |||
0d575f65f9 | |||
34e888f6d6 |
43
README.md
43
README.md
@ -1,24 +1,30 @@
|
|||||||
# Haier hOn
|
# Haier hOn
|
||||||
[](https://github.com/hacs/integration)
|
[](https://hacs.xyz)
|
||||||
|
[](https://github.com/Andre0512/hon/releases/latest)
|
||||||
Home Assistant component supporting devices of Haier's mobile app **hOn**.
|

|
||||||
|
[](https://analytics.home-assistant.io/)
|
||||||
|
Home Assistant integration for Haier hOn: support for Haier/Candy/Hoover home appliances like washing machines.
|
||||||
## Supported Appliances
|
## Supported Appliances
|
||||||
- Washing Machine
|
|
||||||
- Tumble Dryer
|
- Tumble Dryer
|
||||||
|
- Washer Dryer
|
||||||
|
- Washing Machine
|
||||||
|
- Oven
|
||||||
|
|
||||||
## Installation
|
## Installation
|
||||||
#### Installing via HACS
|
**Method 1:** [](https://my.home-assistant.io/redirect/hacs_repository/?owner=Andre0512&repository=hon&category=integration)
|
||||||
1. You need to have installed [HACS](https://hacs.xyz/)
|
|
||||||
2. Go to HACS->Integrations
|
**Method 2:** [HACS](https://hacs.xyz/) > Integrations > Add Integration > **Haier hOn** > Install
|
||||||
3. Add this repo (`https://github.com/Andre0512/hon.git`) into your HACS custom repositories
|
|
||||||
4. Search for Haier hOn and Download it
|
**Method 3:** Manually copy `hon` folder from [latest release](https://github.com/Andre0512/hon/releases/latest) to `config/custom_components` folder.
|
||||||
5. Restart your HomeAssistant
|
|
||||||
6. Go to Settings->Devices & Services
|
_Restart Home Assistant_
|
||||||
7. Shift reload your browser
|
|
||||||
8. Click Add Integration
|
## Configuration
|
||||||
9. Search for Haier hOn
|
|
||||||
10. Type your username used in the hOn App and hit submit
|
**Method 1**: [](https://my.home-assistant.io/redirect/config_flow_start/?domain=hon)
|
||||||
|
|
||||||
|
**Method 2**: Settings > Devices & Services > Add Integration > **Haier hOn**
|
||||||
|
_If the integration is not in the list, you need to clear the browser cache._
|
||||||
|
|
||||||
## Contribute
|
## Contribute
|
||||||
Any kind of contribution is welcome!
|
Any kind of contribution is welcome!
|
||||||
@ -62,12 +68,15 @@ Any kind of contribution is welcome!
|
|||||||
5. Create a [pull request](https://github.com/Andre0512/hon/pulls)
|
5. Create a [pull request](https://github.com/Andre0512/hon/pulls)
|
||||||
|
|
||||||
#### Tips and Tricks
|
#### Tips and Tricks
|
||||||
- If you want to have some states humanreadable, have a look at the `translation_key` parameter of the `EntityDescription`
|
- If you want to have some states humanreadable, have a look at the `translation_key` parameter of the `EntityDescription`.
|
||||||
- If you need to implement some more logic, create a pull request to the underlying library. There we collect special requirements in the `appliances` directory.
|
- If you need to implement some more logic, create a pull request to the underlying library. There we collect special requirements in the `appliances` directory.
|
||||||
|
- Use [pyhOn](https://github.com/Andre0512/pyhOn)s translate command to read out the official translations
|
||||||
|
|
||||||
## Tested Devices
|
## Tested Devices
|
||||||
- Haier WD90-B14TEAM5
|
- Haier WD90-B14TEAM5
|
||||||
- Haier HD80-A3959
|
- Haier HD80-A3959
|
||||||
|
- Haier HWO60SM2F3XH
|
||||||
|
- Hoover H-WASH 500
|
||||||
|
|
||||||
## About this Repo
|
## About this Repo
|
||||||
The existing integrations missed some features from the app I liked to have in HomeAssistant.
|
The existing integrations missed some features from the app I liked to have in HomeAssistant.
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import logging
|
import logging
|
||||||
|
|
||||||
import voluptuous as vol
|
import voluptuous as vol
|
||||||
from pyhon import HonConnection
|
from pyhon import Hon
|
||||||
|
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import CONF_EMAIL, CONF_PASSWORD
|
from homeassistant.const import CONF_EMAIL, CONF_PASSWORD
|
||||||
@ -28,8 +28,7 @@ CONFIG_SCHEMA = vol.Schema(
|
|||||||
|
|
||||||
async def async_setup_entry(hass: HomeAssistantType, entry: ConfigEntry):
|
async def async_setup_entry(hass: HomeAssistantType, entry: ConfigEntry):
|
||||||
session = aiohttp_client.async_get_clientsession(hass)
|
session = aiohttp_client.async_get_clientsession(hass)
|
||||||
hon = HonConnection(entry.data["email"], entry.data["password"], session)
|
hon = await Hon(entry.data["email"], entry.data["password"], session=session).create()
|
||||||
await hon.setup()
|
|
||||||
hass.data.setdefault(DOMAIN, {})
|
hass.data.setdefault(DOMAIN, {})
|
||||||
hass.data[DOMAIN][entry.unique_id] = hon
|
hass.data[DOMAIN][entry.unique_id] = hon
|
||||||
hass.data[DOMAIN]["coordinators"] = {}
|
hass.data[DOMAIN]["coordinators"] = {}
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import logging
|
import logging
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
|
|
||||||
from pyhon import HonConnection
|
from pyhon import Hon
|
||||||
|
|
||||||
from homeassistant.components.binary_sensor import BinarySensorEntityDescription, BinarySensorDeviceClass, \
|
from homeassistant.components.binary_sensor import BinarySensorEntityDescription, BinarySensorDeviceClass, \
|
||||||
BinarySensorEntity
|
BinarySensorEntity
|
||||||
@ -27,15 +27,22 @@ BINARY_SENSORS: dict[str, tuple[HonBinarySensorEntityDescription, ...]] = {
|
|||||||
"WM": (
|
"WM": (
|
||||||
HonBinarySensorEntityDescription(
|
HonBinarySensorEntityDescription(
|
||||||
key="attributes.lastConnEvent.category",
|
key="attributes.lastConnEvent.category",
|
||||||
name="Connection",
|
name="Remote Control",
|
||||||
device_class=BinarySensorDeviceClass.CONNECTIVITY,
|
device_class=BinarySensorDeviceClass.CONNECTIVITY,
|
||||||
on_value="CONNECTED",
|
on_value="CONNECTED",
|
||||||
|
icon="mdi:remote"
|
||||||
),
|
),
|
||||||
HonBinarySensorEntityDescription(
|
HonBinarySensorEntityDescription(
|
||||||
key="doorLockStatus",
|
key="doorLockStatus",
|
||||||
|
name="Door Lock",
|
||||||
|
device_class=BinarySensorDeviceClass.LOCK,
|
||||||
|
on_value="0",
|
||||||
|
),
|
||||||
|
HonBinarySensorEntityDescription(
|
||||||
|
key="doorStatus",
|
||||||
name="Door",
|
name="Door",
|
||||||
device_class=BinarySensorDeviceClass.DOOR,
|
device_class=BinarySensorDeviceClass.DOOR,
|
||||||
on_value="0",
|
on_value="1",
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
"TD": (
|
"TD": (
|
||||||
@ -51,15 +58,75 @@ BINARY_SENSORS: dict[str, tuple[HonBinarySensorEntityDescription, ...]] = {
|
|||||||
device_class=BinarySensorDeviceClass.DOOR,
|
device_class=BinarySensorDeviceClass.DOOR,
|
||||||
on_value="1",
|
on_value="1",
|
||||||
),
|
),
|
||||||
)
|
),
|
||||||
|
"WD": (
|
||||||
|
HonBinarySensorEntityDescription(
|
||||||
|
key="attributes.lastConnEvent.category",
|
||||||
|
name="Remote Control",
|
||||||
|
device_class=BinarySensorDeviceClass.CONNECTIVITY,
|
||||||
|
on_value="CONNECTED",
|
||||||
|
icon="mdi:remote"
|
||||||
|
),
|
||||||
|
HonBinarySensorEntityDescription(
|
||||||
|
key="startProgram.prewash",
|
||||||
|
name="Pre Wash",
|
||||||
|
),
|
||||||
|
HonBinarySensorEntityDescription(
|
||||||
|
key="extraRinse1",
|
||||||
|
name="Extra Rinse 1",
|
||||||
|
),
|
||||||
|
HonBinarySensorEntityDescription(
|
||||||
|
key="extraRinse2",
|
||||||
|
name="Extra Rinse 2",
|
||||||
|
),
|
||||||
|
HonBinarySensorEntityDescription(
|
||||||
|
key="extraRinse3",
|
||||||
|
name="Extra Rinse 3",
|
||||||
|
),
|
||||||
|
HonBinarySensorEntityDescription(
|
||||||
|
key="goodNight",
|
||||||
|
name="Good Night Mode",
|
||||||
|
),
|
||||||
|
HonBinarySensorEntityDescription(
|
||||||
|
key="acquaplus",
|
||||||
|
name="Acqua Plus",
|
||||||
|
),
|
||||||
|
HonBinarySensorEntityDescription(
|
||||||
|
key="anticrease",
|
||||||
|
name="Anti-Crease",
|
||||||
|
),
|
||||||
|
),
|
||||||
|
"OV": (
|
||||||
|
HonBinarySensorEntityDescription(
|
||||||
|
key="attributes.lastConnEvent.category",
|
||||||
|
name="Online",
|
||||||
|
device_class=BinarySensorDeviceClass.CONNECTIVITY,
|
||||||
|
on_value="CONNECTED",
|
||||||
|
icon="mdi:wifi"
|
||||||
|
),
|
||||||
|
HonBinarySensorEntityDescription(
|
||||||
|
key="attributes.parameters.remoteCtrValid",
|
||||||
|
name="On",
|
||||||
|
device_class=BinarySensorDeviceClass.CONNECTIVITY,
|
||||||
|
on_value="1",
|
||||||
|
icon="mdi:remote"
|
||||||
|
),
|
||||||
|
HonBinarySensorEntityDescription(
|
||||||
|
key="attributes.parameters.onOffStatus",
|
||||||
|
name="On",
|
||||||
|
device_class=BinarySensorDeviceClass.RUNNING,
|
||||||
|
on_value="1",
|
||||||
|
icon="mdi:power-cycle"
|
||||||
|
),
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
||||||
hon: HonConnection = hass.data[DOMAIN][entry.unique_id]
|
hon: Hon = hass.data[DOMAIN][entry.unique_id]
|
||||||
coordinators = hass.data[DOMAIN]["coordinators"]
|
coordinators = hass.data[DOMAIN]["coordinators"]
|
||||||
appliances = []
|
appliances = []
|
||||||
for device in hon.devices:
|
for device in hon.appliances:
|
||||||
if device.mac_address in coordinators:
|
if device.mac_address in coordinators:
|
||||||
coordinator = hass.data[DOMAIN]["coordinators"][device.mac_address]
|
coordinator = hass.data[DOMAIN]["coordinators"][device.mac_address]
|
||||||
else:
|
else:
|
||||||
@ -70,7 +137,7 @@ async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> Non
|
|||||||
if descriptions := BINARY_SENSORS.get(device.appliance_type):
|
if descriptions := BINARY_SENSORS.get(device.appliance_type):
|
||||||
for description in descriptions:
|
for description in descriptions:
|
||||||
if not device.get(description.key):
|
if not device.get(description.key):
|
||||||
_LOGGER.info("Can't setup %s", description.key)
|
_LOGGER.warning("[%s] Can't setup %s", device.appliance_type, description.key)
|
||||||
continue
|
continue
|
||||||
appliances.extend([
|
appliances.extend([
|
||||||
HonBinarySensorEntity(hass, coordinator, entry, device, description)]
|
HonBinarySensorEntity(hass, coordinator, entry, device, description)]
|
||||||
|
@ -1,33 +1,32 @@
|
|||||||
from pyhon import HonConnection
|
|
||||||
from pyhon.device import HonDevice
|
|
||||||
|
|
||||||
from homeassistant.components.button import ButtonEntityDescription, ButtonEntity
|
from homeassistant.components.button import ButtonEntityDescription, ButtonEntity
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
|
from pyhon import Hon
|
||||||
|
from pyhon.appliance import HonAppliance
|
||||||
|
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN
|
||||||
from .hon import HonCoordinator, HonEntity
|
from .hon import HonCoordinator, HonEntity
|
||||||
|
|
||||||
BUTTONS: dict[str, tuple[ButtonEntityDescription, ...]] = {
|
BUTTONS: dict[str, tuple[ButtonEntityDescription, ...]] = {
|
||||||
"WM": (
|
"OV": (
|
||||||
# ButtonEntityDescription(
|
ButtonEntityDescription(
|
||||||
# key="pauseProgram",
|
key="startProgram",
|
||||||
# name="Pause Program",
|
name="Start Program",
|
||||||
# icon="mdi:pause",
|
icon="mdi:power-cycle",
|
||||||
# ),
|
),
|
||||||
# ButtonEntityDescription(
|
ButtonEntityDescription(
|
||||||
# key="resumeProgram",
|
key="stopProgram",
|
||||||
# name="Resume Program",
|
name="Stop Program",
|
||||||
# icon="mdi:play-pause",
|
icon="mdi:power-off",
|
||||||
# ),
|
),
|
||||||
),
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
||||||
hon: HonConnection = hass.data[DOMAIN][entry.unique_id]
|
hon: Hon = hass.data[DOMAIN][entry.unique_id]
|
||||||
coordinators = hass.data[DOMAIN]["coordinators"]
|
coordinators = hass.data[DOMAIN]["coordinators"]
|
||||||
appliances = []
|
appliances = []
|
||||||
for device in hon.devices:
|
for device in hon.appliances:
|
||||||
if device.mac_address in coordinators:
|
if device.mac_address in coordinators:
|
||||||
coordinator = hass.data[DOMAIN]["coordinators"][device.mac_address]
|
coordinator = hass.data[DOMAIN]["coordinators"][device.mac_address]
|
||||||
else:
|
else:
|
||||||
@ -47,7 +46,7 @@ async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> Non
|
|||||||
|
|
||||||
|
|
||||||
class HonButtonEntity(HonEntity, ButtonEntity):
|
class HonButtonEntity(HonEntity, ButtonEntity):
|
||||||
def __init__(self, hass, coordinator, entry, device: HonDevice, description) -> None:
|
def __init__(self, hass, coordinator, entry, device: HonAppliance, description) -> None:
|
||||||
super().__init__(hass, entry, coordinator, device)
|
super().__init__(hass, entry, coordinator, device)
|
||||||
|
|
||||||
self._coordinator = coordinator
|
self._coordinator = coordinator
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
import logging
|
import logging
|
||||||
from datetime import timedelta
|
from datetime import timedelta
|
||||||
|
|
||||||
from pyhon.device import HonDevice
|
from pyhon.appliance import HonAppliance
|
||||||
|
|
||||||
from homeassistant.helpers.entity import DeviceInfo
|
from homeassistant.helpers.entity import DeviceInfo
|
||||||
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
from homeassistant.helpers.update_coordinator import CoordinatorEntity
|
||||||
@ -15,7 +15,7 @@ _LOGGER = logging.getLogger(__name__)
|
|||||||
class HonEntity(CoordinatorEntity):
|
class HonEntity(CoordinatorEntity):
|
||||||
_attr_has_entity_name = True
|
_attr_has_entity_name = True
|
||||||
|
|
||||||
def __init__(self, hass, entry, coordinator, device: HonDevice) -> None:
|
def __init__(self, hass, entry, coordinator, device: HonAppliance) -> None:
|
||||||
super().__init__(coordinator)
|
super().__init__(coordinator)
|
||||||
|
|
||||||
self._hon = hass.data[DOMAIN][entry.unique_id]
|
self._hon = hass.data[DOMAIN][entry.unique_id]
|
||||||
@ -36,7 +36,7 @@ class HonEntity(CoordinatorEntity):
|
|||||||
|
|
||||||
|
|
||||||
class HonCoordinator(DataUpdateCoordinator):
|
class HonCoordinator(DataUpdateCoordinator):
|
||||||
def __init__(self, hass, device: HonDevice):
|
def __init__(self, hass, device: HonAppliance):
|
||||||
"""Initialize my coordinator."""
|
"""Initialize my coordinator."""
|
||||||
super().__init__(hass, _LOGGER, name=device.mac_address, update_interval=timedelta(seconds=30))
|
super().__init__(hass, _LOGGER, name=device.mac_address, update_interval=timedelta(seconds=30))
|
||||||
self._device = device
|
self._device = device
|
||||||
|
@ -6,7 +6,7 @@
|
|||||||
"documentation": "https://github.com/Andre0512/hon/",
|
"documentation": "https://github.com/Andre0512/hon/",
|
||||||
"iot_class": "cloud_polling",
|
"iot_class": "cloud_polling",
|
||||||
"issue_tracker": "https://github.com/Andre0512/hon/issues",
|
"issue_tracker": "https://github.com/Andre0512/hon/issues",
|
||||||
"requirements": ["pyhOn==0.4.1"],
|
"requirements": ["pyhOn==0.6.2"],
|
||||||
"version": "0.3.0"
|
"version": "0.5.0-beta.3"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from pyhon import HonConnection
|
from pyhon import Hon
|
||||||
from pyhon.parameter import HonParameterRange
|
from pyhon.parameter import HonParameterRange
|
||||||
|
|
||||||
from homeassistant.components.number import (
|
from homeassistant.components.number import (
|
||||||
@ -8,7 +8,7 @@ from homeassistant.components.number import (
|
|||||||
NumberEntityDescription,
|
NumberEntityDescription,
|
||||||
)
|
)
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import UnitOfTime
|
from homeassistant.const import UnitOfTime, UnitOfTemperature
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
from homeassistant.helpers.entity import EntityCategory
|
from homeassistant.helpers.entity import EntityCategory
|
||||||
|
|
||||||
@ -74,14 +74,47 @@ NUMBERS: dict[str, tuple[NumberEntityDescription, ...]] = {
|
|||||||
entity_category=EntityCategory.CONFIG
|
entity_category=EntityCategory.CONFIG
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
"WD": (
|
||||||
|
NumberEntityDescription(
|
||||||
|
key="startProgram.delayTime",
|
||||||
|
name="Delay Time",
|
||||||
|
icon="mdi:timer-plus",
|
||||||
|
entity_category=EntityCategory.CONFIG,
|
||||||
|
native_unit_of_measurement=UnitOfTime.MINUTES
|
||||||
|
),
|
||||||
|
),
|
||||||
|
"OV": (
|
||||||
|
NumberEntityDescription(
|
||||||
|
key="startProgram.delayTime",
|
||||||
|
name="Delay time",
|
||||||
|
icon="mdi:timer-plus",
|
||||||
|
entity_category=EntityCategory.CONFIG,
|
||||||
|
native_unit_of_measurement=UnitOfTime.MINUTES
|
||||||
|
),
|
||||||
|
NumberEntityDescription(
|
||||||
|
key="startProgram.tempSel",
|
||||||
|
name="Target Temperature",
|
||||||
|
entity_category=EntityCategory.CONFIG,
|
||||||
|
icon="mdi:thermometer",
|
||||||
|
native_unit_of_measurement=UnitOfTemperature.CELSIUS
|
||||||
|
),
|
||||||
|
|
||||||
|
NumberEntityDescription(
|
||||||
|
key="startProgram.prTime",
|
||||||
|
name="Program Duration",
|
||||||
|
entity_category=EntityCategory.CONFIG,
|
||||||
|
icon="mdi:timelapse",
|
||||||
|
native_unit_of_measurement=UnitOfTime.MINUTES
|
||||||
|
),
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
||||||
hon: HonConnection = hass.data[DOMAIN][entry.unique_id]
|
hon: Hon = hass.data[DOMAIN][entry.unique_id]
|
||||||
coordinators = hass.data[DOMAIN]["coordinators"]
|
coordinators = hass.data[DOMAIN]["coordinators"]
|
||||||
appliances = []
|
appliances = []
|
||||||
for device in hon.devices:
|
for device in hon.appliances:
|
||||||
if device.mac_address in coordinators:
|
if device.mac_address in coordinators:
|
||||||
coordinator = hass.data[DOMAIN]["coordinators"][device.mac_address]
|
coordinator = hass.data[DOMAIN]["coordinators"][device.mac_address]
|
||||||
else:
|
else:
|
||||||
@ -91,6 +124,8 @@ async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> Non
|
|||||||
|
|
||||||
if descriptions := NUMBERS.get(device.appliance_type):
|
if descriptions := NUMBERS.get(device.appliance_type):
|
||||||
for description in descriptions:
|
for description in descriptions:
|
||||||
|
if not device.settings.get(description.key):
|
||||||
|
continue
|
||||||
appliances.extend([
|
appliances.extend([
|
||||||
HonNumberEntity(hass, coordinator, entry, device, description)]
|
HonNumberEntity(hass, coordinator, entry, device, description)]
|
||||||
)
|
)
|
||||||
|
@ -1,18 +1,22 @@
|
|||||||
from __future__ import annotations
|
from __future__ import annotations
|
||||||
|
|
||||||
from pyhon import HonConnection
|
import logging
|
||||||
from pyhon.device import HonDevice
|
|
||||||
|
from pyhon import Hon
|
||||||
|
from pyhon.appliance import HonAppliance
|
||||||
from pyhon.parameter import HonParameterFixed
|
from pyhon.parameter import HonParameterFixed
|
||||||
|
|
||||||
from homeassistant.components.select import SelectEntity, SelectEntityDescription
|
from homeassistant.components.select import SelectEntity, SelectEntityDescription
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import UnitOfTemperature, REVOLUTIONS_PER_MINUTE
|
from homeassistant.const import UnitOfTemperature, UnitOfTime, REVOLUTIONS_PER_MINUTE
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
from homeassistant.helpers.entity import EntityCategory
|
from homeassistant.helpers.entity import EntityCategory
|
||||||
|
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN
|
||||||
from .hon import HonEntity, HonCoordinator
|
from .hon import HonEntity, HonCoordinator
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
SELECTS = {
|
SELECTS = {
|
||||||
"WM": (
|
"WM": (
|
||||||
SelectEntityDescription(
|
SelectEntityDescription(
|
||||||
@ -43,15 +47,42 @@ SELECTS = {
|
|||||||
entity_category=EntityCategory.CONFIG,
|
entity_category=EntityCategory.CONFIG,
|
||||||
translation_key="programs"
|
translation_key="programs"
|
||||||
),
|
),
|
||||||
)
|
SelectEntityDescription(
|
||||||
|
key="startProgram.dryTimeMM",
|
||||||
|
name="Time",
|
||||||
|
entity_category=EntityCategory.CONFIG,
|
||||||
|
icon="mdi:timer",
|
||||||
|
unit_of_measurement=UnitOfTime.MINUTES
|
||||||
|
),
|
||||||
|
),
|
||||||
|
"WD": (
|
||||||
|
SelectEntityDescription(
|
||||||
|
key="startProgram.program",
|
||||||
|
name="Program",
|
||||||
|
entity_category=EntityCategory.CONFIG,
|
||||||
|
translation_key="programs"
|
||||||
|
),
|
||||||
|
),
|
||||||
|
"OV": (
|
||||||
|
SelectEntityDescription(
|
||||||
|
key="startProgram.program",
|
||||||
|
name="Program",
|
||||||
|
entity_category=EntityCategory.CONFIG,
|
||||||
|
),
|
||||||
|
SelectEntityDescription(
|
||||||
|
key="startProgram.preheatStatus",
|
||||||
|
name="Preheat",
|
||||||
|
entity_category=EntityCategory.CONFIG
|
||||||
|
),
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
||||||
hon: HonConnection = hass.data[DOMAIN][entry.unique_id]
|
hon: Hon = hass.data[DOMAIN][entry.unique_id]
|
||||||
coordinators = hass.data[DOMAIN]["coordinators"]
|
coordinators = hass.data[DOMAIN]["coordinators"]
|
||||||
appliances = []
|
appliances = []
|
||||||
for device in hon.devices:
|
for device in hon.appliances:
|
||||||
if device.mac_address in coordinators:
|
if device.mac_address in coordinators:
|
||||||
coordinator = hass.data[DOMAIN]["coordinators"][device.mac_address]
|
coordinator = hass.data[DOMAIN]["coordinators"][device.mac_address]
|
||||||
else:
|
else:
|
||||||
@ -61,7 +92,7 @@ async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> Non
|
|||||||
|
|
||||||
if descriptions := SELECTS.get(device.appliance_type):
|
if descriptions := SELECTS.get(device.appliance_type):
|
||||||
for description in descriptions:
|
for description in descriptions:
|
||||||
if not device.get(description.key):
|
if not device.settings.get(description.key):
|
||||||
continue
|
continue
|
||||||
appliances.extend([
|
appliances.extend([
|
||||||
HonSelectEntity(hass, coordinator, entry, device, description)]
|
HonSelectEntity(hass, coordinator, entry, device, description)]
|
||||||
@ -70,7 +101,7 @@ async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> Non
|
|||||||
|
|
||||||
|
|
||||||
class HonSelectEntity(HonEntity, SelectEntity):
|
class HonSelectEntity(HonEntity, SelectEntity):
|
||||||
def __init__(self, hass, coordinator, entry, device: HonDevice, description) -> None:
|
def __init__(self, hass, coordinator, entry, device: HonAppliance, description) -> None:
|
||||||
super().__init__(hass, entry, coordinator, device)
|
super().__init__(hass, entry, coordinator, device)
|
||||||
|
|
||||||
self._coordinator = coordinator
|
self._coordinator = coordinator
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
import logging
|
import logging
|
||||||
|
|
||||||
from pyhon import HonConnection
|
from pyhon import Hon
|
||||||
|
|
||||||
from homeassistant.components.sensor import (
|
from homeassistant.components.sensor import (
|
||||||
SensorEntity,
|
SensorEntity,
|
||||||
@ -9,7 +9,15 @@ from homeassistant.components.sensor import (
|
|||||||
SensorEntityDescription,
|
SensorEntityDescription,
|
||||||
)
|
)
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import UnitOfEnergy, UnitOfVolume, UnitOfMass, UnitOfPower, UnitOfTime
|
from homeassistant.const import (
|
||||||
|
REVOLUTIONS_PER_MINUTE,
|
||||||
|
UnitOfEnergy,
|
||||||
|
UnitOfVolume,
|
||||||
|
UnitOfMass,
|
||||||
|
UnitOfPower,
|
||||||
|
UnitOfTime,
|
||||||
|
UnitOfTemperature
|
||||||
|
)
|
||||||
from homeassistant.core import callback
|
from homeassistant.core import callback
|
||||||
from homeassistant.helpers.entity import EntityCategory
|
from homeassistant.helpers.entity import EntityCategory
|
||||||
from homeassistant.helpers.typing import StateType
|
from homeassistant.helpers.typing import StateType
|
||||||
@ -85,7 +93,7 @@ SENSORS: dict[str, tuple[SensorEntityDescription, ...]] = {
|
|||||||
SensorEntityDescription(
|
SensorEntityDescription(
|
||||||
key="spinSpeed",
|
key="spinSpeed",
|
||||||
name="Spin Speed",
|
name="Spin Speed",
|
||||||
icon="mdi:timer",
|
icon="mdi:speedometer",
|
||||||
state_class=SensorStateClass.MEASUREMENT,
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
native_unit_of_measurement=UnitOfTime.MINUTES,
|
native_unit_of_measurement=UnitOfTime.MINUTES,
|
||||||
),
|
),
|
||||||
@ -141,15 +149,92 @@ SENSORS: dict[str, tuple[SensorEntityDescription, ...]] = {
|
|||||||
icon="mdi:thermometer",
|
icon="mdi:thermometer",
|
||||||
translation_key="tumbledryertemplevel"
|
translation_key="tumbledryertemplevel"
|
||||||
),
|
),
|
||||||
)
|
),
|
||||||
|
"WD": (
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="machMode",
|
||||||
|
name="Machine Status",
|
||||||
|
icon="mdi:information",
|
||||||
|
translation_key="mode"
|
||||||
|
),
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="spinSpeed",
|
||||||
|
name="Spin Speed",
|
||||||
|
icon="mdi:fast-forward-outline",
|
||||||
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
native_unit_of_measurement=REVOLUTIONS_PER_MINUTE,
|
||||||
|
),
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="remainingTimeMM",
|
||||||
|
name="Remaining Time",
|
||||||
|
icon="mdi:timer",
|
||||||
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
native_unit_of_measurement=UnitOfTime.MINUTES,
|
||||||
|
),
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="prCode",
|
||||||
|
name="Current Program",
|
||||||
|
icon="mdi:tumble-dryer",
|
||||||
|
),
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="prPhase",
|
||||||
|
name="Program Phase",
|
||||||
|
icon="mdi:tumble-dryer",
|
||||||
|
),
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="dryLevel",
|
||||||
|
name="Dry level",
|
||||||
|
icon="mdi:hair-dryer",
|
||||||
|
),
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="dirtyLevel",
|
||||||
|
name="Dirt level",
|
||||||
|
icon="mdi:liquid-spot",
|
||||||
|
),
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="steamLevel",
|
||||||
|
name="Steam level",
|
||||||
|
icon="mdi:smoke",
|
||||||
|
),
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="temp",
|
||||||
|
name="Current Temperature",
|
||||||
|
icon="mdi:thermometer",
|
||||||
|
state_class=SensorStateClass.MEASUREMENT,
|
||||||
|
native_unit_of_measurement=UnitOfTemperature.CELSIUS,
|
||||||
|
),
|
||||||
|
),
|
||||||
|
"OV": (
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="remainingTimeMM",
|
||||||
|
name="Remaining Time",
|
||||||
|
icon="mdi:timer",
|
||||||
|
native_unit_of_measurement=UnitOfTime.MINUTES,
|
||||||
|
),
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="delayTime",
|
||||||
|
name="Start Time",
|
||||||
|
icon="mdi:clock-start",
|
||||||
|
),
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="temp",
|
||||||
|
name="Temperature",
|
||||||
|
icon="mdi:thermometer",
|
||||||
|
),
|
||||||
|
SensorEntityDescription(
|
||||||
|
key="tempSel",
|
||||||
|
name="Temperature Selected",
|
||||||
|
icon="mdi:thermometer",
|
||||||
|
),
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
||||||
hon: HonConnection = hass.data[DOMAIN][entry.unique_id]
|
hon: Hon = hass.data[DOMAIN][entry.unique_id]
|
||||||
coordinators = hass.data[DOMAIN]["coordinators"]
|
coordinators = hass.data[DOMAIN]["coordinators"]
|
||||||
appliances = []
|
appliances = []
|
||||||
for device in hon.devices:
|
for device in hon.appliances:
|
||||||
if device.mac_address in coordinators:
|
if device.mac_address in coordinators:
|
||||||
coordinator = hass.data[DOMAIN]["coordinators"][device.mac_address]
|
coordinator = hass.data[DOMAIN]["coordinators"][device.mac_address]
|
||||||
else:
|
else:
|
||||||
@ -160,6 +245,7 @@ async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> Non
|
|||||||
if descriptions := SENSORS.get(device.appliance_type):
|
if descriptions := SENSORS.get(device.appliance_type):
|
||||||
for description in descriptions:
|
for description in descriptions:
|
||||||
if not device.get(description.key):
|
if not device.get(description.key):
|
||||||
|
_LOGGER.warning("[%s] Can't setup %s", device.appliance_type, description.key)
|
||||||
continue
|
continue
|
||||||
appliances.extend([
|
appliances.extend([
|
||||||
HonSensorEntity(hass, coordinator, entry, device, description)]
|
HonSensorEntity(hass, coordinator, entry, device, description)]
|
||||||
|
@ -1,15 +1,19 @@
|
|||||||
|
import logging
|
||||||
|
|
||||||
from dataclasses import dataclass
|
from dataclasses import dataclass
|
||||||
from typing import Any
|
from typing import Any
|
||||||
|
|
||||||
from homeassistant.components.switch import SwitchEntityDescription, SwitchEntity
|
from homeassistant.components.switch import SwitchEntityDescription, SwitchEntity
|
||||||
from homeassistant.config_entries import ConfigEntry
|
from homeassistant.config_entries import ConfigEntry
|
||||||
from homeassistant.const import EntityCategory
|
from homeassistant.const import EntityCategory
|
||||||
from pyhon import HonConnection
|
from pyhon import Hon
|
||||||
from pyhon.device import HonDevice
|
from pyhon.appliance import HonAppliance
|
||||||
|
|
||||||
from .const import DOMAIN
|
from .const import DOMAIN
|
||||||
from .hon import HonCoordinator, HonEntity
|
from .hon import HonCoordinator, HonEntity
|
||||||
|
|
||||||
|
_LOGGER = logging.getLogger(__name__)
|
||||||
|
|
||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class HonSwitchEntityDescriptionMixin:
|
class HonSwitchEntityDescriptionMixin:
|
||||||
@ -19,8 +23,8 @@ class HonSwitchEntityDescriptionMixin:
|
|||||||
|
|
||||||
@dataclass
|
@dataclass
|
||||||
class HonSwitchEntityDescription(HonSwitchEntityDescriptionMixin,
|
class HonSwitchEntityDescription(HonSwitchEntityDescriptionMixin,
|
||||||
SwitchEntityDescription
|
SwitchEntityDescription
|
||||||
):
|
):
|
||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
@ -69,14 +73,30 @@ SWITCHES: dict[str, tuple[HonSwitchEntityDescription, ...]] = {
|
|||||||
turn_off_key="resumeProgram",
|
turn_off_key="resumeProgram",
|
||||||
),
|
),
|
||||||
),
|
),
|
||||||
|
"WD": (
|
||||||
|
HonSwitchEntityDescription(
|
||||||
|
key="active",
|
||||||
|
name="Washing Machine",
|
||||||
|
icon="mdi:washing-machine",
|
||||||
|
turn_on_key="startProgram",
|
||||||
|
turn_off_key="stopProgram",
|
||||||
|
),
|
||||||
|
HonSwitchEntityDescription(
|
||||||
|
key="pause",
|
||||||
|
name="Pause Washing Machine",
|
||||||
|
icon="mdi:pause",
|
||||||
|
turn_on_key="pauseProgram",
|
||||||
|
turn_off_key="resumeProgram",
|
||||||
|
),
|
||||||
|
),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> None:
|
||||||
hon: HonConnection = hass.data[DOMAIN][entry.unique_id]
|
hon: Hon = hass.data[DOMAIN][entry.unique_id]
|
||||||
coordinators = hass.data[DOMAIN]["coordinators"]
|
coordinators = hass.data[DOMAIN]["coordinators"]
|
||||||
appliances = []
|
appliances = []
|
||||||
for device in hon.devices:
|
for device in hon.appliances:
|
||||||
if device.mac_address in coordinators:
|
if device.mac_address in coordinators:
|
||||||
coordinator = hass.data[DOMAIN]["coordinators"][device.mac_address]
|
coordinator = hass.data[DOMAIN]["coordinators"][device.mac_address]
|
||||||
else:
|
else:
|
||||||
@ -90,6 +110,8 @@ async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> Non
|
|||||||
appliances.extend([
|
appliances.extend([
|
||||||
HonSwitchEntity(hass, coordinator, entry, device, description)]
|
HonSwitchEntity(hass, coordinator, entry, device, description)]
|
||||||
)
|
)
|
||||||
|
else:
|
||||||
|
_LOGGER.warning("[%s] Can't setup %s", device.appliance_type, description.key)
|
||||||
|
|
||||||
async_add_entities(appliances)
|
async_add_entities(appliances)
|
||||||
|
|
||||||
@ -97,7 +119,7 @@ async def async_setup_entry(hass, entry: ConfigEntry, async_add_entities) -> Non
|
|||||||
class HonSwitchEntity(HonEntity, SwitchEntity):
|
class HonSwitchEntity(HonEntity, SwitchEntity):
|
||||||
entity_description: HonSwitchEntityDescription
|
entity_description: HonSwitchEntityDescription
|
||||||
|
|
||||||
def __init__(self, hass, coordinator, entry, device: HonDevice, description: HonSwitchEntityDescription) -> None:
|
def __init__(self, hass, coordinator, entry, device: HonAppliance, description: HonSwitchEntityDescription) -> None:
|
||||||
super().__init__(hass, entry, coordinator, device)
|
super().__init__(hass, entry, coordinator, device)
|
||||||
self._coordinator = coordinator
|
self._coordinator = coordinator
|
||||||
self._device = device
|
self._device = device
|
||||||
@ -132,5 +154,3 @@ class HonSwitchEntity(HonEntity, SwitchEntity):
|
|||||||
self.async_write_ha_state()
|
self.async_write_ha_state()
|
||||||
else:
|
else:
|
||||||
await self._device.commands[self.entity_description.turn_off_key].send()
|
await self._device.commands[self.entity_description.turn_off_key].send()
|
||||||
|
|
||||||
|
|
||||||
|
@ -377,6 +377,7 @@
|
|||||||
"single_item_steam": "Single Item + Steam",
|
"single_item_steam": "Single Item + Steam",
|
||||||
"smart_wash": "Smart Wash",
|
"smart_wash": "Smart Wash",
|
||||||
"soft_care": "Soft Care",
|
"soft_care": "Soft Care",
|
||||||
|
"soft_care_steam": "Soft Care + Steam",
|
||||||
"soft_care_steam_title": "Soft Care + Steam",
|
"soft_care_steam_title": "Soft Care + Steam",
|
||||||
"special_39": "Special 39'",
|
"special_39": "Special 39'",
|
||||||
"special_39_full_load": "Special 39'",
|
"special_39_full_load": "Special 39'",
|
||||||
|
@ -377,6 +377,7 @@
|
|||||||
"single_item_steam": "Single Item + Steam",
|
"single_item_steam": "Single Item + Steam",
|
||||||
"smart_wash": "Smart Wash",
|
"smart_wash": "Smart Wash",
|
||||||
"soft_care": "Soft Care",
|
"soft_care": "Soft Care",
|
||||||
|
"soft_care_steam": "Soft Care + Steam",
|
||||||
"soft_care_steam_title": "Soft Care + Steam",
|
"soft_care_steam_title": "Soft Care + Steam",
|
||||||
"special_39": "Special 39'",
|
"special_39": "Special 39'",
|
||||||
"special_39_full_load": "Special 39'",
|
"special_39_full_load": "Special 39'",
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
{
|
{
|
||||||
"name": "Haier hOn",
|
"name": "Haier hOn",
|
||||||
"render_readme": true,
|
|
||||||
"homeassistant": "2023.2.0"
|
"homeassistant": "2023.2.0"
|
||||||
}
|
}
|
||||||
|
36
info.md
Normal file
36
info.md
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
# Haier hOn
|
||||||
|
[](https://github.com/Andre0512/hon/releases/latest)
|
||||||
|

|
||||||
|
[](https://analytics.home-assistant.io/)
|
||||||
|
Support for home appliances of Haier's mobile app hOn.
|
||||||
|
|
||||||
|
## Supported Appliances
|
||||||
|
- Tumble Dryer
|
||||||
|
- Washer Dryer
|
||||||
|
- Washing Machine
|
||||||
|
- Oven
|
||||||
|
|
||||||
|
## Tested Appliances
|
||||||
|
- Haier WD90-B14TEAM5
|
||||||
|
- Haier HD80-A3959
|
||||||
|
- Haier HWO60SM2F3XH
|
||||||
|
- Hoover H-WASH 500
|
||||||
|
|
||||||
|
|
||||||
|
## Configuration
|
||||||
|
|
||||||
|
**Method 1**: [](https://my.home-assistant.io/redirect/config_flow_start/?domain=hon)
|
||||||
|
|
||||||
|
**Method 2**: Settings > Devices & Services > Add Integration > **Haier hOn**
|
||||||
|
_If the integration is not in the list, you need to clear the browser cache._
|
||||||
|
|
||||||
|
|
||||||
|
## Contribute
|
||||||
|
Want to help us to support more appliances?
|
||||||
|
Or add more sensors?
|
||||||
|
Or help with translating?
|
||||||
|
Or beautify some icons or captions?
|
||||||
|
Check out the [project on GitHub](https://github.com/Andre0512/hon), every contribution is welcome!
|
||||||
|
|
||||||
|
|
||||||
|
|
Reference in New Issue
Block a user