145 lines
5.7 KiB
Python
145 lines
5.7 KiB
Python
from appdaemon.entity import Entity
|
|
from appdaemon.plugins.hass.hassapi import Hass
|
|
import json
|
|
|
|
class TapDial(Hass):
|
|
active_entity: Entity
|
|
|
|
def initialize(self):
|
|
self.tap_dial.listen_state(self.handle_state_change)
|
|
|
|
@property
|
|
def tap_dial(self) -> Entity:
|
|
return self.get_entity(self.args['entity'])
|
|
|
|
@property
|
|
def step_size(self) -> int:
|
|
return int(self.args.get('step_size', 25))
|
|
|
|
@property
|
|
def init_brightness(self) -> int:
|
|
return int(self.args.get('init_brightness', 200))
|
|
|
|
@property
|
|
def max_brightness(self) -> int:
|
|
return int(self.args.get('max_brightness', 254))
|
|
|
|
@property
|
|
def active_entity_is_on(self) -> bool:
|
|
return self.active_entity.get_state() == 'on'
|
|
|
|
@property
|
|
def light_brightness(self) -> int:
|
|
return self.active_entity.attributes['brightness']
|
|
|
|
@light_brightness.setter
|
|
def light_brightness(self, val: int):
|
|
return self.active_entity.turn_on(brightness=val)
|
|
|
|
@property
|
|
def light_color(self) -> list[int]:
|
|
return self.active_entity.get_state('rgb_color')
|
|
|
|
@light_color.setter
|
|
def light_color(self, val: list[int]):
|
|
return self.active_entity.turn_on(rgb_color=val)
|
|
|
|
@property
|
|
def light_effect(self) -> str:
|
|
return self.active_entity.get_state('effect')
|
|
|
|
@light_effect.setter
|
|
def light_effect(self, val: str):
|
|
return self.active_entity.turn_on(effect=val)
|
|
|
|
def handle_state_change(self, entity: str, attribute: str, old: str, new: str, **kwargs):
|
|
self.log(f'056 State change: {new}')
|
|
|
|
# Dial actions
|
|
if new.startswith('dial_rotate') and self.active_entity_is_on:
|
|
_, _, dir, speed = new.split('_', 4)
|
|
self.log(f'061 Direction {dir} speed {speed}')
|
|
match speed:
|
|
case 'step':
|
|
rate = self.step_size
|
|
case 'slow':
|
|
rate = 2 * self.step_size
|
|
case 'fast':
|
|
rate = 3 * self.step_size
|
|
val = self.active_entity.attributes['brightness']
|
|
self.log(f'070 Brightness value = {val} Change rate = {rate}')
|
|
if dir == 'right':
|
|
if val < self.max_brightness:
|
|
if (val + rate) > self.max_brightness:
|
|
val = self.max_brightness
|
|
else:
|
|
val += rate
|
|
self.active_entity.turn_on(brightness=val)
|
|
self.log(f'078 Brightness value = {val}')
|
|
else:
|
|
if val > 0:
|
|
if (val - rate) < 0:
|
|
val = 0
|
|
else:
|
|
val -= rate
|
|
self.active_entity.turn_on(brightness=val)
|
|
self.log(f'086 Brightness value = {val}')
|
|
|
|
# Button actions
|
|
elif new.endswith('release'):
|
|
_, n, typ, _ = new.split('_', 4)
|
|
# type will be either press or hold
|
|
self.log(f'092 Button {n} {typ}')
|
|
|
|
if eid := self.args.get(f'button{n}'):
|
|
self.active_entity = self.get_entity(eid)
|
|
self.log(f'096 Set active entity to: {self.active_entity.name}')
|
|
self.log(json.dumps(self.active_entity.get_state('all'), indent=4))
|
|
|
|
domain, entity = eid.split('.')
|
|
self.log(f'099 Curent domain: {domain}, entity: {entity}')
|
|
match domain:
|
|
case 'light':
|
|
# Set the light to maximum brightness if the button is held.
|
|
if typ == 'hold':
|
|
self.active_entity.turn_on(brightness=self.max_brightness)
|
|
self.log(f'105 Set {self.active_entity.friendly_name} to maximum brightness {self.max_brightness}')
|
|
else:
|
|
if self.active_entity_is_on:
|
|
self.active_entity.turn_off()
|
|
self.log(f'109 Turn off {self.active_entity.friendly_name}')
|
|
else:
|
|
self.active_entity.turn_on(brightness=self.init_brightness)
|
|
self.log(f'112 Turn on {self.active_entity.friendly_name} with brightness {self.init_brightness}')
|
|
case 'switch':
|
|
self.active_entity.toggle()
|
|
self.log(f'115 Toggle on/off power to {self.active_entity.friendly_name}')
|
|
|
|
elif switch := self.args.get(f'button{n}'):
|
|
self.active_entity = self.get_entity(switch)
|
|
self.log(f'119 Set active entity to: {self.active_entity.friendly_name}')
|
|
onoff = self.active_entity.get_state()
|
|
self.log(f'114 {self.active_entity.friendly_name} is currently {onoff}')
|
|
self.active_entity.toggle()
|
|
self.log(f'116 Toggle on/off power to {self.active_entity.friendly_name}')
|
|
|
|
# This function was written by chatgpt!
|
|
def hex_to_rgb(self, hex_color: str) -> list:
|
|
|
|
# Remove the "#" if it's included
|
|
hex_color = hex_color.lstrip('#')
|
|
|
|
# Check if the string has a valid length
|
|
if len(hex_color) != 6:
|
|
raise ValueError(f"126 Invalid hex color: {hex_color}. Must be 6 characters long.")
|
|
|
|
try:
|
|
# Split the hex color into its RGB components and convert to integers
|
|
r = int(hex_color[0:2], 16)
|
|
g = int(hex_color[2:4], 16)
|
|
b = int(hex_color[4:6], 16)
|
|
return [r, g, b]
|
|
except ValueError:
|
|
raise ValueError(f"135 Invalid hex color: {hex_color}. Must contain only valid hex digits.")
|
|
|