generalized main - get values from config
This commit is contained in:
parent
823552ce7f
commit
da9373e74d
|
@ -1,4 +1,5 @@
|
||||||
generated_3d
|
generated_3d
|
||||||
|
venv
|
||||||
build
|
build
|
||||||
*.FCStd1
|
*.FCStd1
|
||||||
*.blend1
|
*.blend1
|
|
@ -39,6 +39,13 @@ mouse_led_effect_to_message_id = {
|
||||||
EffectReverseSwipe: 10,
|
EffectReverseSwipe: 10,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
mouse_leds_index_ranges = {
|
||||||
|
TouchButton.RIGHT_FOOT: (0, 6),
|
||||||
|
TouchButton.LEFT_FOOT: (6, 6 + 6),
|
||||||
|
TouchButton.LEFT_EAR: (6 + 6, 6 + 6 + 16),
|
||||||
|
TouchButton.RIGHT_EAR: (6 + 6 + 16, 6 + 6 + 16 + 17),
|
||||||
|
}
|
||||||
|
|
||||||
PREV_BUTTON_LED_MSG = 20
|
PREV_BUTTON_LED_MSG = 20
|
||||||
NEXT_BUTTON_LED_MSG = 21
|
NEXT_BUTTON_LED_MSG = 21
|
||||||
|
|
||||||
|
|
|
@ -1,4 +1,7 @@
|
||||||
|
#!/usr/bin/env python
|
||||||
|
|
||||||
import asyncio
|
import asyncio
|
||||||
|
import sys
|
||||||
import serial_asyncio
|
import serial_asyncio
|
||||||
from led_cmds import (ColorRGBW, ColorHSV, EffectStaticConfig,
|
from led_cmds import (ColorRGBW, ColorHSV, EffectStaticConfig,
|
||||||
EffectRandomTwoColorInterpolationConfig, EffectAlexaSwipeConfig,
|
EffectRandomTwoColorInterpolationConfig, EffectAlexaSwipeConfig,
|
||||||
|
@ -9,134 +12,129 @@ from glob import glob
|
||||||
from copy import deepcopy
|
from copy import deepcopy
|
||||||
import os
|
import os
|
||||||
from hass_client import HomeAssistantClient
|
from hass_client import HomeAssistantClient
|
||||||
|
import argparse
|
||||||
|
from ruamel.yaml import YAML
|
||||||
|
import warnings
|
||||||
|
from pprint import pprint
|
||||||
|
|
||||||
HASS_URL = "https://ha.bauer.tech"
|
yaml = YAML(typ='safe')
|
||||||
HASS_TOKEN = "eyJ0eXAiOiJKV1QiLCJhbGciOiJIUzI1NiJ9.eyJpc3MiOiI3YmY1OTc3NWJhZGI0MzFkOTBiNDBkZDA3OGQ1MTMxNSIsImlhdCI6MTY0MDAzNzI3MSwiZXhwIjoxOTU1Mzk3MjcxfQ.c7RPRP_hxzIwd3xcFTNLz94rOjLQDR0elH8RE-jCDgc"
|
|
||||||
MUSIC_FOLDER = "/home/martin/code/musicmouse/espmusicmouse/host_driver/music"
|
|
||||||
|
|
||||||
audio_player = AudioPlayer()
|
OFF_COLOR = ColorRGBW(0, 0, 0, 0)
|
||||||
current_figure = None
|
|
||||||
last_figure = None
|
|
||||||
|
|
||||||
rfid_token_map = {
|
|
||||||
bytes.fromhex("88041174e9"): "elefant",
|
|
||||||
bytes.fromhex("8804ce7230"): "fuchs",
|
|
||||||
bytes.fromhex("88040d71f0"): "eule",
|
|
||||||
bytes.fromhex("88043c6ede"): "omnom",
|
|
||||||
bytes.fromhex("88040b78ff"): "eichhoernchen",
|
|
||||||
bytes.fromhex("8804bc7444"): "hund",
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def parse_hex_color(color_str: str):
|
def parse_color(color_str: str):
|
||||||
if isinstance(color_str, ColorRGBW):
|
if isinstance(color_str, ColorRGBW):
|
||||||
return color_str
|
return color_str
|
||||||
|
elif color_str.startswith("#"):
|
||||||
color_str = color_str.lstrip('#')
|
color_str = color_str.lstrip('#')
|
||||||
t = tuple(int(color_str[i:i + 2], 16) / 255 for i in (0, 2, 4))
|
t = tuple(int(color_str[i:i + 2], 16) / 255 for i in (0, 2, 4))
|
||||||
return ColorRGBW(*t, 0)
|
return ColorRGBW(*t, 0)
|
||||||
|
elif color_str.startswith("w"):
|
||||||
|
color_str = color_str.lstrip("w")
|
||||||
|
return ColorRGBW(0, 0, 0, int(color_str, 16) / 255)
|
||||||
|
|
||||||
|
|
||||||
class FigureColorCfg:
|
def load_config(config_path):
|
||||||
def __init__(self, primary, secondary, background, accent):
|
with open(os.path.join(config_path, "config.yml")) as cfg_file:
|
||||||
self.primary = parse_hex_color(primary)
|
cfg = yaml.load(cfg_file)
|
||||||
self.secondary = parse_hex_color(secondary)
|
for figure_name, figure_cfg in cfg["figures"].items():
|
||||||
self.background = parse_hex_color(background)
|
figure_cfg["colors"] = [parse_color(c) for c in figure_cfg["colors"]]
|
||||||
self.accent = parse_hex_color(accent)
|
if 'media_files' not in figure_cfg:
|
||||||
|
figure_cfg['media_files'] = sorted(glob(os.path.join(config_path, figure_name)))
|
||||||
|
return cfg
|
||||||
|
|
||||||
|
|
||||||
color_cfg = {
|
class MusicMouseState:
|
||||||
"elefant": FigureColorCfg("#ffff00", "#00c8ff", "#094b46", "#c20099"),
|
def __init__(self, protocol: MusicMouseProtocol):
|
||||||
"fuchs": FigureColorCfg("#F4D35E", "#F95738", "#F95738", "#083d77"),
|
self.current_figure: str = None
|
||||||
"omnom": FigureColorCfg("#005102", "#3bc405", "#005102", "#3bc405"),
|
self.last_figure: str = None
|
||||||
"eichhoernchen": FigureColorCfg("#ff0ada", "#4BC6B9", "#69045a", "#4BC6B9"),
|
self.current_mouse_led_effect = None
|
||||||
"hund": FigureColorCfg("#ffff00", "#00c8ff", "#094b46", "#c20099"),
|
self.current_led_ring_effect = None
|
||||||
"eule": FigureColorCfg("#e5a200", "#f8e300", ColorRGBW(0, 0, 0, 0.2), ColorRGBW(0, 0, 0, 1)),
|
self.protocol: MusicMouseProtocol = protocol
|
||||||
|
self.button_led_brightness = None
|
||||||
|
|
||||||
|
def mouse_led_effect(self, effect_cfg):
|
||||||
|
self.current_mouse_led_effect = effect_cfg
|
||||||
|
self.protocol.mouse_led_effect(effect_cfg)
|
||||||
|
|
||||||
|
def led_ring_effect(self, effect_cfg):
|
||||||
|
self.current_led_ring_effect = effect_cfg
|
||||||
|
self.protocol.led_ring_effect(effect_cfg)
|
||||||
|
|
||||||
|
def button_leds(self, brightness):
|
||||||
|
assert 0 <= brightness <= 1
|
||||||
|
self.protocol.button_background_led_prev(brightness)
|
||||||
|
self.protocol.button_background_led_next(brightness)
|
||||||
|
self.button_led_brightness = brightness
|
||||||
|
|
||||||
|
def reset(self):
|
||||||
|
self.mouse_led_effect(EffectStaticConfig(OFF_COLOR))
|
||||||
|
self.led_ring_effect(EffectStaticConfig(OFF_COLOR))
|
||||||
|
|
||||||
|
def figure_placed(self, figure_state):
|
||||||
|
self.last_figure = self.current_figure
|
||||||
|
self.current_figure = figure_state
|
||||||
|
|
||||||
|
def figure_removed(self):
|
||||||
|
self.last_figure = self.current_figure
|
||||||
|
|
||||||
|
|
||||||
|
class Controller:
|
||||||
|
def __init__(self, protocol, cfg):
|
||||||
|
self.cfg = cfg
|
||||||
|
self.audio_player = AudioPlayer(cfg["general"]["alsa_device"])
|
||||||
|
self.mmstate = MusicMouseState(protocol)
|
||||||
|
|
||||||
|
protocol.register_message_callback(self.on_firmware_msg)
|
||||||
|
|
||||||
|
self.audio_player.on_playlist_end_callback = self._run_off_animation
|
||||||
|
self.playlists = {
|
||||||
|
fig: self.audio_player.create_playlist(fig_cfg['media_files'])
|
||||||
|
for fig, fig_cfg in cfg['figures'].items()
|
||||||
|
}
|
||||||
|
self._rfid_to_figure_name = {
|
||||||
|
bytes.fromhex(figure_cfg["id"]): figure_name
|
||||||
|
for figure_name, figure_cfg in cfg["figures"].items()
|
||||||
}
|
}
|
||||||
|
|
||||||
playlists = {
|
def handle_rfid_event(self, tagid):
|
||||||
fig: audio_player.create_playlist(sorted(glob(os.path.join(MUSIC_FOLDER, fig, "*.mp3"))))
|
|
||||||
for fig in rfid_token_map.values()
|
|
||||||
}
|
|
||||||
|
|
||||||
mouse_leds = {
|
|
||||||
TouchButton.RIGHT_FOOT: (0, 6),
|
|
||||||
TouchButton.LEFT_FOOT: (6, 6 + 6),
|
|
||||||
TouchButton.LEFT_EAR: (6 + 6, 6 + 6 + 16),
|
|
||||||
TouchButton.RIGHT_EAR: (6 + 6 + 16, 6 + 6 + 16 + 17),
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
def run_off_animation(protocol):
|
|
||||||
ring_eff = EffectReverseSwipe()
|
|
||||||
mouse_eff = EffectReverseSwipe()
|
|
||||||
mouse_eff.startPosition = 6 / 45 * 360
|
|
||||||
protocol.led_ring_effect(ring_eff)
|
|
||||||
protocol.mouse_led_effect(mouse_eff)
|
|
||||||
protocol.button_background_led_prev(0)
|
|
||||||
protocol.button_background_led_next(0)
|
|
||||||
|
|
||||||
|
|
||||||
def on_music_end_callback(protocol):
|
|
||||||
run_off_animation(protocol)
|
|
||||||
|
|
||||||
|
|
||||||
def on_rfid(protocol, tagid):
|
|
||||||
global current_figure, last_figure
|
|
||||||
|
|
||||||
if tagid == bytes.fromhex("0000000000"):
|
if tagid == bytes.fromhex("0000000000"):
|
||||||
# Off
|
if self.audio_player.is_playing():
|
||||||
if audio_player.is_playing():
|
self._run_off_animation()
|
||||||
run_off_animation(protocol)
|
self.audio_player.pause()
|
||||||
audio_player.pause()
|
self.mmstate.figure_removed()
|
||||||
#else:
|
elif tagid in self._rfid_to_figure_name:
|
||||||
# protocol.led_ring_effect(EffectStaticConfig(ColorRGBW(0, 0, 0, 0)))
|
figure = self._rfid_to_figure_name[tagid]
|
||||||
# protocol.mouse_led_effect(EffectStaticConfig(ColorRGBW(0, 0, 0, 0)))
|
primary_color, secondary_color, *rest = self.cfg["figures"]["colors"]
|
||||||
last_figure = current_figure
|
self._start_animation(primary_color, secondary_color)
|
||||||
|
self.mmstate.button_leds(self.cfg["general"].get("button_leds_brightness", 0.5))
|
||||||
|
|
||||||
|
if figure in self.playlists:
|
||||||
|
self.audio_player.set_playlist(self.playlists[figure])
|
||||||
|
if self.mmstate.last_figure == figure:
|
||||||
|
self.audio_player.play()
|
||||||
else:
|
else:
|
||||||
figure = rfid_token_map[tagid]
|
self.audio_player.play_from_start()
|
||||||
current_figure = figure
|
|
||||||
ccfg = color_cfg[figure]
|
|
||||||
ring_eff = EffectSwipeAndChange()
|
|
||||||
ring_eff.swipe.primary_color = ccfg.primary
|
|
||||||
ring_eff.swipe.secondary_color = ccfg.secondary
|
|
||||||
ring_eff.swipe.swipe_speed = 180
|
|
||||||
ring_eff.change.color1 = ccfg.primary
|
|
||||||
ring_eff.change.color2 = ccfg.secondary
|
|
||||||
protocol.led_ring_effect(ring_eff)
|
|
||||||
|
|
||||||
mouse_eff = EffectStaticConfig(ccfg.background)
|
self.mmstate.figure_placed(figure)
|
||||||
mouse_eff = deepcopy(ring_eff)
|
|
||||||
mouse_eff.swipe.start_position = 6 / 45 * 360
|
|
||||||
mouse_eff.swipe.bell_curve_width_in_leds = 16
|
|
||||||
mouse_eff.swipe.swipe_speed = 180
|
|
||||||
protocol.mouse_led_effect(mouse_eff)
|
|
||||||
|
|
||||||
protocol.button_background_led_prev(0.3)
|
|
||||||
protocol.button_background_led_next(0.3)
|
|
||||||
|
|
||||||
if figure in playlists:
|
|
||||||
audio_player.set_playlist(playlists[figure])
|
|
||||||
if last_figure == current_figure:
|
|
||||||
audio_player.play()
|
|
||||||
else:
|
else:
|
||||||
audio_player.play_from_start()
|
warnings.warn(f"Unknown figure/tag with id {tagid}")
|
||||||
|
|
||||||
|
def on_firmware_msg(self, _, message):
|
||||||
def on_firmware_msg(protocol: MusicMouseProtocol, message):
|
|
||||||
print("FW msg:", message)
|
print("FW msg:", message)
|
||||||
if isinstance(message, RfidTokenRead):
|
if isinstance(message, RfidTokenRead):
|
||||||
on_rfid(protocol, message.id)
|
self.handle_rfid_event(message.id)
|
||||||
elif isinstance(message, RotaryEncoderEvent):
|
elif isinstance(message, RotaryEncoderEvent):
|
||||||
if audio_player.is_playing():
|
volume_increment = self.cfg["general"].get("volume_increment", 2)
|
||||||
if message.direction == 2:
|
if message.direction == 2:
|
||||||
audio_player.change_volume(2)
|
self.audio_player.change_volume(volume_increment)
|
||||||
elif message.direction == 1:
|
elif message.direction == 1:
|
||||||
audio_player.change_volume(-2)
|
self.audio_player.change_volume(-volume_increment)
|
||||||
elif isinstance(message, ButtonEvent):
|
elif isinstance(message, ButtonEvent):
|
||||||
if message.button == "left" and message.event == "pressed":
|
if message.button == "left" and message.event == "pressed":
|
||||||
audio_player.previous()
|
self.audio_player.previous()
|
||||||
elif message.button == "right" and message.event == "pressed":
|
elif message.button == "right" and message.event == "pressed":
|
||||||
audio_player.next()
|
self.audio_player.next()
|
||||||
elif False and isinstance(message, TouchButtonPress):
|
elif False and isinstance(message, TouchButtonPress):
|
||||||
if current_figure:
|
if current_figure:
|
||||||
ccfg = color_cfg[current_figure]
|
ccfg = color_cfg[current_figure]
|
||||||
|
@ -155,7 +153,8 @@ def on_firmware_msg(protocol: MusicMouseProtocol, message):
|
||||||
if current_figure:
|
if current_figure:
|
||||||
ccfg = color_cfg[current_figure]
|
ccfg = color_cfg[current_figure]
|
||||||
eff_change = EffectRandomTwoColorInterpolationConfig()
|
eff_change = EffectRandomTwoColorInterpolationConfig()
|
||||||
eff_static = EffectStaticConfig(ColorRGBW(0, 0, 0, 0), *mouse_leds[message.touch_button])
|
eff_static = EffectStaticConfig(ColorRGBW(0, 0, 0, 0),
|
||||||
|
*mouse_leds[message.touch_button])
|
||||||
if audio_player.is_playing():
|
if audio_player.is_playing():
|
||||||
eff_static.color = ccfg.primary
|
eff_static.color = ccfg.primary
|
||||||
protocol.mouse_led_effect(eff_static)
|
protocol.mouse_led_effect(eff_static)
|
||||||
|
@ -166,18 +165,48 @@ def on_firmware_msg(protocol: MusicMouseProtocol, message):
|
||||||
eff_change.start_with_existing = True
|
eff_change.start_with_existing = True
|
||||||
protocol.mouse_led_effect(eff_change)
|
protocol.mouse_led_effect(eff_change)
|
||||||
|
|
||||||
|
def _start_animation(self, primary_color, secondary_color):
|
||||||
|
ring_eff = EffectSwipeAndChange()
|
||||||
|
ring_eff.swipe.primary_color = primary_color
|
||||||
|
ring_eff.swipe.secondary_color = secondary_color
|
||||||
|
ring_eff.swipe.swipe_speed = 180
|
||||||
|
ring_eff.change.color1 = primary_color
|
||||||
|
ring_eff.change.color2 = secondary_color
|
||||||
|
self.mmstate.led_ring_effect(ring_eff)
|
||||||
|
|
||||||
|
mouse_eff = deepcopy(ring_eff)
|
||||||
|
mouse_eff.swipe.start_position = 6 / 45 * 360
|
||||||
|
mouse_eff.swipe.bell_curve_width_in_leds = 16
|
||||||
|
mouse_eff.swipe.swipe_speed = 180
|
||||||
|
self.mmstate.mouse_led_effect(mouse_eff)
|
||||||
|
|
||||||
|
def _run_off_animation(self):
|
||||||
|
ring_eff = EffectReverseSwipe()
|
||||||
|
self.mmstate.led_ring_effect(ring_eff)
|
||||||
|
|
||||||
|
mouse_eff = EffectReverseSwipe()
|
||||||
|
mouse_eff.startPosition = 6 / 45 * 360
|
||||||
|
self.mmstate.mouse_led_effect(mouse_eff)
|
||||||
|
|
||||||
|
self.mmstate.button_leds(0)
|
||||||
|
|
||||||
|
|
||||||
|
def main(config_path):
|
||||||
|
cfg = load_config(config_path)
|
||||||
loop = asyncio.get_event_loop()
|
loop = asyncio.get_event_loop()
|
||||||
hass = HomeAssistantClient(HASS_URL, HASS_TOKEN, loop)
|
|
||||||
coro = serial_asyncio.create_serial_connection(loop,
|
coro = serial_asyncio.create_serial_connection(loop,
|
||||||
MusicMouseProtocol,
|
MusicMouseProtocol,
|
||||||
'/dev/ttyUSB0',
|
cfg["general"]["serial_port"],
|
||||||
baudrate=115200)
|
baudrate=115200)
|
||||||
transport, protocol = loop.run_until_complete(coro)
|
transport, protocol = loop.run_until_complete(coro)
|
||||||
protocol.register_message_callback(on_firmware_msg)
|
controller = Controller(protocol, cfg)
|
||||||
|
return controller, loop
|
||||||
|
|
||||||
audio_player.on_playlist_end_callback = lambda: on_music_end_callback(protocol)
|
|
||||||
|
|
||||||
loop.create_task(hass.connect())
|
if __name__ == "__main__":
|
||||||
|
if len(sys.argv) == 2:
|
||||||
|
controller, loop = main(config_path=sys.argv[1])
|
||||||
loop.run_forever()
|
loop.run_forever()
|
||||||
loop.close()
|
loop.close()
|
||||||
|
else:
|
||||||
|
print("Error: run with config file path as first argument")
|
||||||
|
|
|
@ -69,8 +69,9 @@ all_events = (
|
||||||
|
|
||||||
|
|
||||||
class AudioPlayer:
|
class AudioPlayer:
|
||||||
def __init__(self):
|
def __init__(self, alsa_device=None):
|
||||||
self.instance = vlc.Instance()
|
params = ["-A", "alsa", "--alsa-audio-device", alsa_device] if alsa_device else []
|
||||||
|
self.instance = vlc.Instance(*params)
|
||||||
self.media_list_player = self.instance.media_list_player_new()
|
self.media_list_player = self.instance.media_list_player_new()
|
||||||
self.media_player = self.media_list_player.get_media_player()
|
self.media_player = self.media_list_player.get_media_player()
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue