|
1
|
|
|
import signal |
|
2
|
|
|
from colorsys import hsv_to_rgb, rgb_to_hsv |
|
3
|
|
|
from threading import Thread |
|
4
|
|
|
|
|
5
|
|
|
from pyhap.accessory import Accessory, Bridge |
|
6
|
|
|
from pyhap.accessory_driver import AccessoryDriver |
|
7
|
|
|
from pyhap.const import CATEGORY_LIGHTBULB |
|
8
|
|
|
|
|
9
|
|
|
|
|
10
|
|
|
class HomeKitLight(Accessory): |
|
11
|
|
|
category = CATEGORY_LIGHTBULB |
|
12
|
|
|
|
|
13
|
|
|
def __init__(self, fixture, *args, **kwargs): |
|
14
|
|
|
|
|
15
|
|
|
super().__init__(*args, **kwargs) |
|
16
|
|
|
|
|
17
|
|
|
serv_light = self.add_preload_service( |
|
18
|
|
|
'Lightbulb', chars=['On', 'Hue', 'Saturation', 'Brightness']) |
|
19
|
|
|
|
|
20
|
|
|
# Set our instance variables |
|
21
|
|
|
self.fixture = fixture |
|
22
|
|
|
self.accessory_state = self.get_state() # State of On/Off |
|
23
|
|
|
self.hue = self.get_hue() # Hue Value 0 - 360 Homekit API |
|
24
|
|
|
self.saturation = self.get_saturation() # Saturation Values 0 - 100 Homekit API |
|
25
|
|
|
self.brightness = self.get_brightness() # Brightness value 0 - 100 Homekit API |
|
26
|
|
|
|
|
27
|
|
|
# Configure our callbacks |
|
28
|
|
|
self.char_on = serv_light.configure_char( |
|
29
|
|
|
'On', setter_callback=self.set_state, getter_callback=self.get_state) |
|
30
|
|
|
self.char_hue = serv_light.configure_char( |
|
31
|
|
|
'Hue', setter_callback=self.set_hue, getter_callback=self.get_hue) |
|
32
|
|
|
self.char_saturation = serv_light.configure_char( |
|
33
|
|
|
'Saturation', setter_callback=self.set_saturation, getter_callback=self.get_saturation) |
|
34
|
|
|
self.char_on = serv_light.configure_char( |
|
35
|
|
|
'Brightness', setter_callback=self.set_brightness, getter_callback=self.get_brightness) |
|
36
|
|
|
|
|
37
|
|
|
def set_state(self, value): |
|
38
|
|
|
self.accessory_state = value |
|
39
|
|
|
if value == 1: # On |
|
40
|
|
|
self.set_brightness(100) |
|
41
|
|
|
else: # Off |
|
42
|
|
|
self.set_brightness(0) |
|
43
|
|
|
|
|
44
|
|
|
def get_state(self): |
|
45
|
|
|
self.accessory_state = 1 if self.get_brightness() > 0 else 0 |
|
46
|
|
|
return self.accessory_state |
|
47
|
|
|
|
|
48
|
|
|
def set_color(self): |
|
49
|
|
|
h = self.hue / 360 |
|
50
|
|
|
s = self.saturation / 100 |
|
51
|
|
|
r, g, b = hsv_to_rgb(h, s, 1) |
|
52
|
|
|
self.fixture.color([r * 255, g * 255, b * 255]) |
|
53
|
|
|
|
|
54
|
|
|
def get_color(self): |
|
55
|
|
|
return self.fixture.get_color()[:3] |
|
56
|
|
|
|
|
57
|
|
|
def set_brightness(self, value): |
|
58
|
|
|
self.accessory_state = 1 if value > 0 else 0 |
|
59
|
|
|
self.brightness = value |
|
60
|
|
|
self.fixture.dim(self.brightness * 255 / 100) |
|
61
|
|
|
|
|
62
|
|
|
def get_brightness(self): |
|
63
|
|
|
return self.fixture.get_channel_value(self.fixture.get_channel_id("dimmer"))[0] * 100 / 255 |
|
64
|
|
|
|
|
65
|
|
|
def set_hue(self, value): |
|
66
|
|
|
self.hue = value |
|
67
|
|
|
self.set_color() |
|
68
|
|
|
|
|
69
|
|
|
def get_hue(self): |
|
70
|
|
|
h, s, v = rgb_to_hsv(*self.get_color()) |
|
71
|
|
|
return h * 360 |
|
72
|
|
|
|
|
73
|
|
|
def set_saturation(self, value): |
|
74
|
|
|
self.saturation = value |
|
75
|
|
|
self.set_color() |
|
76
|
|
|
|
|
77
|
|
|
def get_saturation(self): |
|
78
|
|
|
h, s, v = rgb_to_hsv(*self.get_color()) |
|
79
|
|
|
return s * 100 |
|
80
|
|
|
|
|
81
|
|
|
|
|
82
|
|
|
def run(controller): |
|
83
|
|
|
start_port = 51826 |
|
84
|
|
|
driver = AccessoryDriver(port=start_port) |
|
85
|
|
|
bridge = Bridge(driver, 'PyDMXControl') |
|
86
|
|
|
|
|
87
|
|
|
for i, fixture in enumerate(controller.get_all_fixtures()): |
|
88
|
|
|
print(fixture.name) |
|
89
|
|
|
bridge.add_accessory(HomeKitLight(fixture, driver, fixture.name)) |
|
90
|
|
|
|
|
91
|
|
|
signal.signal(signal.SIGTERM, driver.signal_handler) |
|
92
|
|
|
driver.add_accessory(accessory=bridge) |
|
93
|
|
|
|
|
94
|
|
|
thread = Thread(target=driver.start) |
|
95
|
|
|
thread.daemon = True |
|
96
|
|
|
thread.start() |
|
97
|
|
|
|