2019-02-13 20:21:14 +00:00
|
|
|
"""Support for ThinkingCleaner switches."""
|
2016-04-10 23:59:53 +00:00
|
|
|
from datetime import timedelta
|
2019-11-26 18:22:57 +00:00
|
|
|
import time
|
|
|
|
|
2020-03-09 11:56:26 +00:00
|
|
|
from pythinkingcleaner import Discovery, ThinkingCleaner
|
|
|
|
import voluptuous as vol
|
2016-04-10 23:59:53 +00:00
|
|
|
|
2018-07-18 09:54:27 +00:00
|
|
|
from homeassistant import util
|
2020-03-09 11:56:26 +00:00
|
|
|
from homeassistant.components.switch import PLATFORM_SCHEMA
|
|
|
|
from homeassistant.const import CONF_HOST, STATE_OFF, STATE_ON
|
|
|
|
import homeassistant.helpers.config_validation as cv
|
2016-04-10 23:59:53 +00:00
|
|
|
from homeassistant.helpers.entity import ToggleEntity
|
|
|
|
|
|
|
|
MIN_TIME_BETWEEN_SCANS = timedelta(seconds=10)
|
|
|
|
MIN_TIME_BETWEEN_FORCED_SCANS = timedelta(milliseconds=100)
|
|
|
|
|
|
|
|
MIN_TIME_TO_WAIT = timedelta(seconds=5)
|
|
|
|
MIN_TIME_TO_LOCK_UPDATE = 5
|
|
|
|
|
|
|
|
SWITCH_TYPES = {
|
2019-07-31 19:25:30 +00:00
|
|
|
"clean": ["Clean", None, None],
|
|
|
|
"dock": ["Dock", None, None],
|
|
|
|
"find": ["Find", None, None],
|
2016-04-10 23:59:53 +00:00
|
|
|
}
|
|
|
|
|
2020-03-09 11:56:26 +00:00
|
|
|
PLATFORM_SCHEMA = PLATFORM_SCHEMA.extend({vol.Optional(CONF_HOST): cv.string})
|
|
|
|
|
2016-04-10 23:59:53 +00:00
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
def setup_platform(hass, config, add_entities, discovery_info=None):
|
2017-05-02 16:18:47 +00:00
|
|
|
"""Set up the ThinkingCleaner platform."""
|
2020-03-09 11:56:26 +00:00
|
|
|
host = config.get(CONF_HOST)
|
|
|
|
if host:
|
|
|
|
devices = [ThinkingCleaner(host, "unknown")]
|
|
|
|
else:
|
|
|
|
discovery = Discovery()
|
|
|
|
devices = discovery.discover()
|
2016-04-10 23:59:53 +00:00
|
|
|
|
|
|
|
@util.Throttle(MIN_TIME_BETWEEN_SCANS, MIN_TIME_BETWEEN_FORCED_SCANS)
|
|
|
|
def update_devices():
|
|
|
|
"""Update all devices."""
|
|
|
|
for device_object in devices:
|
|
|
|
device_object.update()
|
|
|
|
|
|
|
|
dev = []
|
|
|
|
for device in devices:
|
2016-07-08 01:54:16 +00:00
|
|
|
for type_name in SWITCH_TYPES:
|
2019-07-31 19:25:30 +00:00
|
|
|
dev.append(ThinkingCleanerSwitch(device, type_name, update_devices))
|
2016-04-10 23:59:53 +00:00
|
|
|
|
2018-08-24 14:37:30 +00:00
|
|
|
add_entities(dev)
|
2016-04-10 23:59:53 +00:00
|
|
|
|
|
|
|
|
|
|
|
class ThinkingCleanerSwitch(ToggleEntity):
|
|
|
|
"""ThinkingCleaner Switch (dock, clean, find me)."""
|
|
|
|
|
|
|
|
def __init__(self, tc_object, switch_type, update_devices):
|
|
|
|
"""Initialize the ThinkingCleaner."""
|
|
|
|
self.type = switch_type
|
|
|
|
|
|
|
|
self._update_devices = update_devices
|
|
|
|
self._tc_object = tc_object
|
2019-07-31 19:25:30 +00:00
|
|
|
self._state = self._tc_object.is_cleaning if switch_type == "clean" else False
|
2016-04-10 23:59:53 +00:00
|
|
|
self.lock = False
|
|
|
|
self.last_lock_time = None
|
|
|
|
self.graceful_state = False
|
|
|
|
|
|
|
|
def lock_update(self):
|
|
|
|
"""Lock the update since TC clean takes some time to update."""
|
|
|
|
if self.is_update_locked():
|
|
|
|
return
|
|
|
|
self.lock = True
|
|
|
|
self.last_lock_time = time.time()
|
|
|
|
|
|
|
|
def reset_update_lock(self):
|
|
|
|
"""Reset the update lock."""
|
|
|
|
self.lock = False
|
|
|
|
self.last_lock_time = None
|
|
|
|
|
|
|
|
def set_graceful_lock(self, state):
|
|
|
|
"""Set the graceful state."""
|
|
|
|
self.graceful_state = state
|
|
|
|
self.reset_update_lock()
|
|
|
|
self.lock_update()
|
|
|
|
|
|
|
|
def is_update_locked(self):
|
|
|
|
"""Check if the update method is locked."""
|
|
|
|
if self.last_lock_time is None:
|
|
|
|
return False
|
|
|
|
|
|
|
|
if time.time() - self.last_lock_time >= MIN_TIME_TO_LOCK_UPDATE:
|
|
|
|
self.last_lock_time = None
|
|
|
|
return False
|
|
|
|
|
|
|
|
return True
|
|
|
|
|
|
|
|
@property
|
|
|
|
def name(self):
|
|
|
|
"""Return the name of the sensor."""
|
2020-04-04 21:09:34 +00:00
|
|
|
return f"{self._tc_object.name} {SWITCH_TYPES[self.type][0]}"
|
2016-04-10 23:59:53 +00:00
|
|
|
|
|
|
|
@property
|
|
|
|
def is_on(self):
|
|
|
|
"""Return true if device is on."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if self.type == "clean":
|
|
|
|
return (
|
|
|
|
self.graceful_state
|
|
|
|
if self.is_update_locked()
|
|
|
|
else self._tc_object.is_cleaning
|
|
|
|
)
|
2016-04-10 23:59:53 +00:00
|
|
|
|
|
|
|
return False
|
|
|
|
|
|
|
|
def turn_on(self, **kwargs):
|
|
|
|
"""Turn the device on."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if self.type == "clean":
|
2016-04-10 23:59:53 +00:00
|
|
|
self.set_graceful_lock(True)
|
|
|
|
self._tc_object.start_cleaning()
|
2019-07-31 19:25:30 +00:00
|
|
|
elif self.type == "dock":
|
2016-04-10 23:59:53 +00:00
|
|
|
self._tc_object.dock()
|
2019-07-31 19:25:30 +00:00
|
|
|
elif self.type == "find":
|
2016-04-10 23:59:53 +00:00
|
|
|
self._tc_object.find_me()
|
|
|
|
|
|
|
|
def turn_off(self, **kwargs):
|
|
|
|
"""Turn the device off."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if self.type == "clean":
|
2016-04-10 23:59:53 +00:00
|
|
|
self.set_graceful_lock(False)
|
|
|
|
self._tc_object.stop_cleaning()
|
|
|
|
|
|
|
|
def update(self):
|
|
|
|
"""Update the switch state (Only for clean)."""
|
2019-07-31 19:25:30 +00:00
|
|
|
if self.type == "clean" and not self.is_update_locked():
|
2016-04-10 23:59:53 +00:00
|
|
|
self._tc_object.update()
|
2019-07-31 19:25:30 +00:00
|
|
|
self._state = STATE_ON if self._tc_object.is_cleaning else STATE_OFF
|