2019-02-13 20:21:14 +00:00
|
|
|
"""Support for Ecovacs Ecovacs Vaccums."""
|
2022-01-03 15:26:14 +00:00
|
|
|
from __future__ import annotations
|
|
|
|
|
2018-08-20 15:42:53 +00:00
|
|
|
import logging
|
|
|
|
|
2019-11-26 02:01:48 +00:00
|
|
|
import sucks
|
|
|
|
|
2018-08-20 15:42:53 +00:00
|
|
|
from homeassistant.components.vacuum import (
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_BATTERY,
|
|
|
|
SUPPORT_CLEAN_SPOT,
|
|
|
|
SUPPORT_FAN_SPEED,
|
|
|
|
SUPPORT_LOCATE,
|
|
|
|
SUPPORT_RETURN_HOME,
|
|
|
|
SUPPORT_SEND_COMMAND,
|
|
|
|
SUPPORT_STATUS,
|
|
|
|
SUPPORT_STOP,
|
|
|
|
SUPPORT_TURN_OFF,
|
|
|
|
SUPPORT_TURN_ON,
|
2020-04-26 00:11:08 +00:00
|
|
|
VacuumEntity,
|
2019-07-31 19:25:30 +00:00
|
|
|
)
|
2022-01-03 15:26:14 +00:00
|
|
|
from homeassistant.core import HomeAssistant
|
|
|
|
from homeassistant.helpers.entity_platform import AddEntitiesCallback
|
2018-08-20 15:42:53 +00:00
|
|
|
from homeassistant.helpers.icon import icon_for_battery_level
|
2022-01-03 15:26:14 +00:00
|
|
|
from homeassistant.helpers.typing import ConfigType, DiscoveryInfoType
|
2018-08-20 15:42:53 +00:00
|
|
|
|
2019-03-21 05:56:46 +00:00
|
|
|
from . import ECOVACS_DEVICES
|
|
|
|
|
2018-08-20 15:42:53 +00:00
|
|
|
_LOGGER = logging.getLogger(__name__)
|
|
|
|
|
|
|
|
SUPPORT_ECOVACS = (
|
2019-07-31 19:25:30 +00:00
|
|
|
SUPPORT_BATTERY
|
|
|
|
| SUPPORT_RETURN_HOME
|
|
|
|
| SUPPORT_CLEAN_SPOT
|
|
|
|
| SUPPORT_STOP
|
|
|
|
| SUPPORT_TURN_OFF
|
|
|
|
| SUPPORT_TURN_ON
|
|
|
|
| SUPPORT_LOCATE
|
|
|
|
| SUPPORT_STATUS
|
|
|
|
| SUPPORT_SEND_COMMAND
|
|
|
|
| SUPPORT_FAN_SPEED
|
|
|
|
)
|
|
|
|
|
|
|
|
ATTR_ERROR = "error"
|
|
|
|
ATTR_COMPONENT_PREFIX = "component_"
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
|
2022-01-03 15:26:14 +00:00
|
|
|
def setup_platform(
|
|
|
|
hass: HomeAssistant,
|
|
|
|
config: ConfigType,
|
|
|
|
add_entities: AddEntitiesCallback,
|
|
|
|
discovery_info: DiscoveryInfoType | None = None,
|
|
|
|
) -> None:
|
2018-08-20 15:42:53 +00:00
|
|
|
"""Set up the Ecovacs vacuums."""
|
|
|
|
vacuums = []
|
|
|
|
for device in hass.data[ECOVACS_DEVICES]:
|
|
|
|
vacuums.append(EcovacsVacuum(device))
|
2020-01-05 12:09:17 +00:00
|
|
|
_LOGGER.debug("Adding Ecovacs Vacuums to Home Assistant: %s", vacuums)
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities(vacuums, True)
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
|
2020-04-26 00:11:08 +00:00
|
|
|
class EcovacsVacuum(VacuumEntity):
|
2018-08-20 15:42:53 +00:00
|
|
|
"""Ecovacs Vacuums such as Deebot."""
|
|
|
|
|
|
|
|
def __init__(self, device):
|
|
|
|
"""Initialize the Ecovacs Vacuum."""
|
|
|
|
self.device = device
|
|
|
|
self.device.connect_and_wait_until_ready()
|
2020-04-07 19:06:05 +00:00
|
|
|
if self.device.vacuum.get("nick") is not None:
|
2020-02-24 16:47:52 +00:00
|
|
|
self._name = str(self.device.vacuum["nick"])
|
2018-09-18 06:30:20 +00:00
|
|
|
else:
|
2018-08-20 15:42:53 +00:00
|
|
|
# In case there is no nickname defined, use the device id
|
2020-02-24 16:47:52 +00:00
|
|
|
self._name = str(format(self.device.vacuum["did"]))
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
self._fan_speed = None
|
|
|
|
self._error = None
|
|
|
|
_LOGGER.debug("Vacuum initialized: %s", self.name)
|
|
|
|
|
|
|
|
async def async_added_to_hass(self) -> None:
|
|
|
|
"""Set up the event listeners now that hass is ready."""
|
2019-07-31 19:25:30 +00:00
|
|
|
self.device.statusEvents.subscribe(lambda _: self.schedule_update_ha_state())
|
|
|
|
self.device.batteryEvents.subscribe(lambda _: self.schedule_update_ha_state())
|
|
|
|
self.device.lifespanEvents.subscribe(lambda _: self.schedule_update_ha_state())
|
2018-08-20 15:42:53 +00:00
|
|
|
self.device.errorEvents.subscribe(self.on_error)
|
|
|
|
|
|
|
|
def on_error(self, error):
|
|
|
|
"""Handle an error event from the robot.
|
|
|
|
|
|
|
|
This will not change the entity's state. If the error caused the state
|
|
|
|
to change, that will come through as a separate on_status event
|
|
|
|
"""
|
2019-07-31 19:25:30 +00:00
|
|
|
if error == "no_error":
|
2018-08-20 15:42:53 +00:00
|
|
|
self._error = None
|
|
|
|
else:
|
|
|
|
self._error = error
|
|
|
|
|
2019-07-31 19:25:30 +00:00
|
|
|
self.hass.bus.fire(
|
|
|
|
"ecovacs_error", {"entity_id": self.entity_id, "error": error}
|
|
|
|
)
|
2018-08-20 15:42:53 +00:00
|
|
|
self.schedule_update_ha_state()
|
|
|
|
|
|
|
|
@property
|
|
|
|
def should_poll(self) -> bool:
|
|
|
|
"""Return True if entity has to be polled for state."""
|
|
|
|
return False
|
|
|
|
|
|
|
|
@property
|
|
|
|
def unique_id(self) -> str:
|
|
|
|
"""Return an unique ID."""
|
2020-04-07 19:06:05 +00:00
|
|
|
return self.device.vacuum.get("did")
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
|
|
|
"""Return true if vacuum is currently cleaning."""
|
|
|
|
return self.device.is_cleaning
|
|
|
|
|
|
|
|
@property
|
|
|
|
def is_charging(self):
|
|
|
|
"""Return true if vacuum is currently charging."""
|
|
|
|
return self.device.is_charging
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the device."""
|
|
|
|
return self._name
|
|
|
|
|
|
|
|
@property
|
|
|
|
def supported_features(self):
|
|
|
|
"""Flag vacuum cleaner robot features that are supported."""
|
|
|
|
return SUPPORT_ECOVACS
|
|
|
|
|
|
|
|
@property
|
|
|
|
def status(self):
|
|
|
|
"""Return the status of the vacuum cleaner."""
|
|
|
|
return self.device.vacuum_status
|
|
|
|
|
|
|
|
def return_to_base(self, **kwargs):
|
|
|
|
"""Set the vacuum cleaner to return to the dock."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-11-26 02:01:48 +00:00
|
|
|
self.device.run(sucks.Charge())
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def battery_icon(self):
|
|
|
|
"""Return the battery icon for the vacuum cleaner."""
|
|
|
|
return icon_for_battery_level(
|
2019-07-31 19:25:30 +00:00
|
|
|
battery_level=self.battery_level, charging=self.is_charging
|
|
|
|
)
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def battery_level(self):
|
|
|
|
"""Return the battery level of the vacuum cleaner."""
|
|
|
|
if self.device.battery_status is not None:
|
|
|
|
return self.device.battery_status * 100
|
|
|
|
|
|
|
|
return super().battery_level
|
|
|
|
|
|
|
|
@property
|
|
|
|
def fan_speed(self):
|
|
|
|
"""Return the fan speed of the vacuum cleaner."""
|
|
|
|
return self.device.fan_speed
|
|
|
|
|
|
|
|
@property
|
|
|
|
def fan_speed_list(self):
|
|
|
|
"""Get the list of available fan speed steps of the vacuum cleaner."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-11-26 02:01:48 +00:00
|
|
|
return [sucks.FAN_SPEED_NORMAL, sucks.FAN_SPEED_HIGH]
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
def turn_on(self, **kwargs):
|
|
|
|
"""Turn the vacuum on and start cleaning."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-11-26 02:01:48 +00:00
|
|
|
self.device.run(sucks.Clean())
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
def turn_off(self, **kwargs):
|
|
|
|
"""Turn the vacuum off stopping the cleaning and returning home."""
|
|
|
|
self.return_to_base()
|
|
|
|
|
|
|
|
def stop(self, **kwargs):
|
|
|
|
"""Stop the vacuum cleaner."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-11-26 02:01:48 +00:00
|
|
|
self.device.run(sucks.Stop())
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
def clean_spot(self, **kwargs):
|
|
|
|
"""Perform a spot clean-up."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-11-26 02:01:48 +00:00
|
|
|
self.device.run(sucks.Spot())
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
def locate(self, **kwargs):
|
|
|
|
"""Locate the vacuum cleaner."""
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-11-26 02:01:48 +00:00
|
|
|
self.device.run(sucks.PlaySound())
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
def set_fan_speed(self, fan_speed, **kwargs):
|
|
|
|
"""Set fan speed."""
|
|
|
|
if self.is_on:
|
2019-07-31 19:25:30 +00:00
|
|
|
|
2019-11-26 02:01:48 +00:00
|
|
|
self.device.run(sucks.Clean(mode=self.device.clean_status, speed=fan_speed))
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
def send_command(self, command, params=None, **kwargs):
|
|
|
|
"""Send a command to a vacuum cleaner."""
|
2019-11-26 02:01:48 +00:00
|
|
|
self.device.run(sucks.VacBotCommand(command, params))
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
@property
|
2021-03-11 15:51:03 +00:00
|
|
|
def extra_state_attributes(self):
|
2018-08-20 15:42:53 +00:00
|
|
|
"""Return the device-specific state attributes of this vacuum."""
|
|
|
|
data = {}
|
|
|
|
data[ATTR_ERROR] = self._error
|
|
|
|
|
|
|
|
for key, val in self.device.components.items():
|
|
|
|
attr_name = ATTR_COMPONENT_PREFIX + key
|
2018-09-23 07:43:01 +00:00
|
|
|
data[attr_name] = int(val * 100)
|
2018-08-20 15:42:53 +00:00
|
|
|
|
|
|
|
return data
|