voicemeeter-compact/vmcompact/channels.py

349 lines
11 KiB
Python
Raw Normal View History

2023-06-26 13:52:24 +01:00
import logging
2022-04-11 18:35:28 +01:00
import tkinter as tk
from tkinter import ttk
2022-04-11 18:35:28 +01:00
from . import builders
from .data import _base_values, _configuration
2022-04-11 18:35:28 +01:00
2023-06-26 13:52:24 +01:00
logger = logging.getLogger(__name__)
2022-04-11 18:35:28 +01:00
class ChannelLabelFrame(ttk.LabelFrame):
2022-04-11 18:35:28 +01:00
"""Base class for a single channel"""
def __init__(self, parent, index, id):
super().__init__(parent)
self.parent = parent
2022-04-11 18:35:28 +01:00
self.index = index
self.id = id
2023-06-26 13:52:24 +01:00
self.logger = logger.getChild(self.__class__.__name__)
self.styletable = self.parent.parent.styletable
self.builder = builders.ChannelLabelFrameBuilder(self, index, id)
self.builder.setup()
self.builder.add_progressbar()
self.builder.add_scale()
self.builder.add_mute_button()
self.builder.add_conf_button()
self.builder.add_gain_label()
self.sync_params()
self.sync_labels()
self.grid_configure()
2022-04-11 18:35:28 +01:00
self.configbuilder = builders.MainFrameBuilder(self.parent.parent)
2022-04-11 18:35:28 +01:00
@property
def identifier(self):
return self.id
@property
def target(self):
"""returns the current interface"""
return self.parent.target
2022-04-11 18:35:28 +01:00
def getter(self, param):
2023-06-26 13:52:24 +01:00
try:
2022-04-11 18:35:28 +01:00
return getattr(self.target, param)
2023-06-26 13:52:24 +01:00
except AttributeError as e:
self.logger(f"{type(e).__name__}: {e}")
2022-04-11 18:35:28 +01:00
def setter(self, param, value):
2023-06-26 13:52:24 +01:00
if param in dir(self.target): # avoid calling getattr (with hasattr)
2022-04-11 18:35:28 +01:00
setattr(self.target, param, value)
def scale_callback(self, *args):
"""callback function for scale widget"""
2023-06-26 13:52:24 +01:00
val = round(self.gain.get(), 1)
self.setter("gain", val)
self.gainlabel.set(val)
2022-04-11 18:35:28 +01:00
def toggle_mute(self, *args):
self.target.mute = self.mute.get()
if not _configuration.themes_enabled:
self.styletable.configure(
2022-04-11 18:35:28 +01:00
f"{self.identifier}Mute{self.index}.TButton",
background=f'{"red" if self.mute.get() else "white"}',
)
def reset_gain(self, *args):
self.setter("gain", 0)
self.gain.set(0)
self.gainlabel.set(self.gain.get())
2022-04-11 18:35:28 +01:00
def scale_press(self, *args):
self.after(1, self.remove_events)
def remove_events(self):
self.parent.target.event.remove("pdirty")
self.parent.target.event.remove("ldirty")
2022-04-11 18:35:28 +01:00
def scale_release(self, *args):
_base_values.run_update = False
self.after(1, self.add_events)
def add_events(self):
self.parent.target.event.add("pdirty")
self.parent.target.event.add("ldirty")
self.after(500, self.resume_updates)
def pause_updates(self, func, *args):
"""function wrapper, adds a 50ms delay on updates"""
_base_values.run_update = False
func(*args)
self.after(50, self.resume_updates)
def resume_updates(self):
_base_values.run_update = True
2022-04-11 18:35:28 +01:00
def _on_mousewheel(self, event):
self.gain.set(
round(
self.gain.get()
+ (
_configuration.mwscroll_step
if event.delta > 0
else -_configuration.mwscroll_step
),
1,
)
)
if self.gain.get() > 12:
self.gain.set(12)
elif self.gain.get() < -60:
self.gain.set(-60)
self.setter("gain", self.gain.get())
self.gainlabel.set(round(self.gain.get(), 1))
2022-04-11 18:35:28 +01:00
def open_config(self):
if self.conf.get():
self.configbuilder.create_configframe(self.identifier, self.index, self.id)
else:
self.parent.parent.config_frame.teardown()
if not _configuration.themes_enabled:
self.styletable.configure(
f"{self.identifier}Conf{self.index}.TButton",
background=f'{"yellow" if self.conf.get() else "white"}',
)
2022-04-11 18:35:28 +01:00
def on_update(self, subject):
if subject == "ldirty":
self.upd_levels()
elif subject == "pdirty":
self.sync_params()
elif subject == "labelframe":
self.after(5, self.sync_labels)
def sync_params(self):
"""sync parameter states, update button colours"""
self.gain.set(self.getter("gain"))
self.gainlabel.set(round(self.gain.get(), 1))
self.mute.set(self.getter("mute"))
if not _configuration.themes_enabled:
self.styletable.configure(
f"{self.identifier}Mute{self.index}.TButton",
background=f'{"red" if self.mute.get() else "white"}',
)
def sync_labels(self):
"""sync labelframes according to label text"""
2022-04-11 18:35:28 +01:00
retval = self.getter("label")
if self.parent.label_cache[self.id][self.index] != retval:
self.parent.label_cache[self.id][self.index] = retval
if len(retval) > 10:
retval = f"{retval[:8]}.."
if not retval:
self.parent.columnconfigure(self.index, minsize=0)
self.parent.parent.subject.remove(self)
self.grid_remove()
else:
self.parent.parent.subject.add(self)
self.grid()
self.configure(text=retval)
2022-04-11 18:35:28 +01:00
def grid_configure(self):
self.grid(padx=_configuration.channel_xpadding, sticky=(tk.N, tk.S))
2022-04-11 18:35:28 +01:00
[
child.grid_configure(padx=1, pady=1, sticky=(tk.W, tk.E))
for child in self.winfo_children()
if isinstance(child, ttk.Checkbutton)
]
[
child.grid_configure(padx=1, pady=1, sticky=(tk.N, tk.S))
for child in self.winfo_children()
if isinstance(child, ttk.Progressbar) or isinstance(child, ttk.Scale)
]
class Strip(ChannelLabelFrame):
"""Concrete class representing a single strip"""
2022-04-11 18:35:28 +01:00
def __init__(self, parent, index, id):
super().__init__(parent, index, id)
if index <= parent.phys_in:
self.level_offset = index * 2
else:
self.level_offset = parent.phys_in * 2 + (index - parent.phys_in) * 8
2022-04-11 18:35:28 +01:00
@property
def target(self):
"""returns the strip class for this labelframe in the current interface"""
2022-04-11 18:35:28 +01:00
_target = super(Strip, self).target
return getattr(_target, self.identifier)[self.index]
def upd_levels(self):
"""
Updates level values.
"""
if self.index < self.parent.parent.kind.num_strip:
if self.target.levels.is_updated:
val = max(self.target.levels.prefader)
2022-10-05 23:39:51 +01:00
self.level.set(
(0 if self.mute.get() else 72 + val - 12 + self.gain.get())
)
2022-04-11 18:35:28 +01:00
class Bus(ChannelLabelFrame):
2022-04-11 18:35:28 +01:00
"""Concrete bus class representing a single bus"""
def __init__(self, parent, index, id):
super().__init__(parent, index, id)
self.level_offset = index * 8
2022-04-11 18:35:28 +01:00
@property
def target(self):
"""returns the bus class for this labelframe in the current interface"""
2022-04-11 18:35:28 +01:00
_target = super(Bus, self).target
return getattr(_target, self.identifier)[self.index]
def upd_levels(self):
if self.index < self.parent.parent.kind.num_bus:
if self.target.levels.is_updated or self.level.get() != -118:
val = max(self.target.levels.all)
2022-10-05 23:39:51 +01:00
self.level.set((0 if self.mute.get() else 72 + val - 12))
2022-04-11 18:35:28 +01:00
class ChannelFrame(ttk.Frame):
def init(self, parent, id):
2022-04-11 18:35:28 +01:00
super().__init__(parent)
self.parent = parent
self.id = id
2022-04-11 18:35:28 +01:00
self.phys_in, self.virt_in = parent.kind.ins
self.phys_out, self.virt_out = parent.kind.outs
self.label_cache = {
"strip": [""] * (self.phys_in + self.virt_in),
"bus": [""] * (self.phys_out + self.virt_out),
}
self.parent.subject.add(self)
self.update_labels()
2022-04-11 18:35:28 +01:00
@property
def target(self):
"""returns the current interface"""
return self.parent.target
2022-04-11 18:35:28 +01:00
@property
def identifier(self):
return self.id
2022-04-11 18:35:28 +01:00
@property
def labelframes(self):
"""returns a tuple of current channel labelframe addresses"""
return tuple(
frame
for frame in self.winfo_children()
if isinstance(frame, ttk.LabelFrame)
)
def update_labels(self):
for labelframe in self.labelframes:
labelframe.on_update("labelframe")
def on_update(self, subject):
if subject == "pdirty":
self.update_labels()
def grid_configure(self):
2022-04-11 18:35:28 +01:00
[
self.columnconfigure(i, minsize=_configuration.channel_width)
for i, _ in enumerate(self.labelframes)
2022-04-11 18:35:28 +01:00
]
[self.rowconfigure(0, minsize=100) for i, _ in enumerate(self.labelframes)]
def teardown(self):
[self.parent.subject.remove(frame) for frame in self.labelframes]
self.parent.subject.remove(self)
self.destroy()
setattr(self.parent, f"{self.identifier}_frame", None)
def _make_channelframe(parent, identifier):
"""
Creates a Channel Frame class of type strip or bus
"""
phys_in, virt_in = parent.kind.ins
phys_out, virt_out = parent.kind.outs
2022-04-11 18:35:28 +01:00
def init_labels(self):
"""
Grids each labelframe, grid_removes any without a label
"""
2022-04-11 18:35:28 +01:00
for i, labelframe in enumerate(
getattr(self, "strips" if identifier == "strip" else "buses")
):
labelframe.grid(row=0, column=i)
label = labelframe.target.label
if not label:
2022-04-11 18:35:28 +01:00
self.columnconfigure(i, minsize=0)
labelframe.grid_remove()
self.label_cache[identifier][i] = label
def init_strip(self, *args, **kwargs):
self.init(parent, identifier)
self.strips = tuple(
Strip(self, i, identifier) for i in range(phys_in + virt_in)
)
self.grid(row=0, column=0, sticky=(tk.W))
self.grid_configure()
init_labels(self)
def init_bus(self, *args, **kwargs):
self.init(parent, identifier)
self.buses = tuple(Bus(self, i, identifier) for i in range(phys_out + virt_out))
if _configuration.extended:
if _configuration.extends_horizontal:
self.grid(row=0, column=2, sticky=(tk.W))
2022-04-11 18:35:28 +01:00
else:
self.grid(row=2, column=0, sticky=(tk.W))
else:
self.grid(row=0, column=0)
self.grid_configure()
init_labels(self)
if identifier == "strip":
CHANNELFRAME_cls = type(
f"ChannelFrame{identifier.capitalize()}",
(ChannelFrame,),
{
"__init__": init_strip,
},
)
else:
CHANNELFRAME_cls = type(
f"ChannelFrame{identifier.capitalize()}",
(ChannelFrame,),
{
"__init__": init_bus,
},
)
return CHANNELFRAME_cls(parent)