2023-06-29 17:15:03 +01:00
|
|
|
import logging
|
2022-04-11 18:35:28 +01:00
|
|
|
import tkinter as tk
|
2023-06-26 16:08:58 +01:00
|
|
|
from functools import cached_property
|
2022-06-16 23:53:28 +01:00
|
|
|
from pathlib import Path
|
2023-07-11 08:35:23 +01:00
|
|
|
from tkinter import messagebox, ttk
|
2022-04-11 18:35:28 +01:00
|
|
|
from typing import NamedTuple
|
|
|
|
|
2023-07-11 08:35:23 +01:00
|
|
|
import voicemeeterlib
|
|
|
|
|
2022-05-10 20:34:29 +01:00
|
|
|
from .builders import MainFrameBuilder
|
2023-06-26 16:08:58 +01:00
|
|
|
from .configurations import loader
|
2023-07-11 08:35:23 +01:00
|
|
|
from .data import _base_values, _configuration, _kinds_all, get_configuration
|
2023-06-26 13:55:50 +01:00
|
|
|
from .errors import VMCompactError
|
2022-04-11 18:35:28 +01:00
|
|
|
from .menu import Menus
|
2022-06-16 23:53:28 +01:00
|
|
|
from .subject import Subject
|
2022-04-11 18:35:28 +01:00
|
|
|
|
2023-06-29 17:15:03 +01:00
|
|
|
logger = logging.getLogger(__name__)
|
|
|
|
|
2022-04-11 18:35:28 +01:00
|
|
|
|
|
|
|
class App(tk.Tk):
|
2022-05-10 20:34:29 +01:00
|
|
|
"""App mainframe"""
|
|
|
|
|
2022-04-11 18:35:28 +01:00
|
|
|
@classmethod
|
|
|
|
def make(cls, kind: NamedTuple):
|
|
|
|
"""
|
2022-05-10 20:34:29 +01:00
|
|
|
Factory function for App.
|
2022-04-11 18:35:28 +01:00
|
|
|
|
2022-05-10 20:34:29 +01:00
|
|
|
Returns an App class of a kind.
|
2022-04-11 18:35:28 +01:00
|
|
|
"""
|
2022-05-10 20:34:29 +01:00
|
|
|
|
2022-04-11 18:35:28 +01:00
|
|
|
APP_cls = type(
|
2025-01-15 20:56:37 +00:00
|
|
|
f'Voicemeeter{kind}.Compact',
|
2022-04-11 18:35:28 +01:00
|
|
|
(cls,),
|
|
|
|
{
|
2025-01-15 20:56:37 +00:00
|
|
|
'kind': kind,
|
2022-04-11 18:35:28 +01:00
|
|
|
},
|
|
|
|
)
|
|
|
|
return APP_cls
|
|
|
|
|
|
|
|
def __init__(self, vmr):
|
|
|
|
super().__init__()
|
2023-06-29 17:15:03 +01:00
|
|
|
self.logger = logger.getChild(self.__class__.__name__)
|
2022-05-10 20:34:29 +01:00
|
|
|
self._vmr = vmr
|
2025-01-15 20:56:37 +00:00
|
|
|
self._vmr.event.add(['pdirty', 'ldirty'])
|
2023-09-07 08:39:20 +01:00
|
|
|
self.subject = Subject()
|
|
|
|
self.start_updates()
|
2023-06-26 13:55:50 +01:00
|
|
|
self._vmr.init_thread()
|
2025-01-15 20:56:37 +00:00
|
|
|
for pn in (
|
|
|
|
Path(__file__).parent.resolve() / 'img' / 'cat.ico',
|
|
|
|
Path.cwd() / '_internal' / 'img' / 'cat.ico',
|
|
|
|
):
|
|
|
|
if pn.is_file():
|
|
|
|
self.iconbitmap(str(pn))
|
|
|
|
break
|
2022-05-10 20:34:29 +01:00
|
|
|
self.minsize(275, False)
|
2023-06-26 16:08:58 +01:00
|
|
|
self._configs = None
|
2025-01-15 20:56:37 +00:00
|
|
|
self.protocol('WM_DELETE_WINDOW', self.on_close_window)
|
|
|
|
self.menu = self['menu'] = Menus(self, vmr)
|
2022-04-11 18:35:28 +01:00
|
|
|
self.styletable = ttk.Style()
|
2022-06-17 17:53:46 +01:00
|
|
|
if _configuration.config:
|
|
|
|
vmr.apply_config(_configuration.config)
|
2022-04-11 18:35:28 +01:00
|
|
|
|
2022-05-10 20:34:29 +01:00
|
|
|
self.build_app()
|
2022-04-11 18:35:28 +01:00
|
|
|
|
2025-01-15 20:56:37 +00:00
|
|
|
self.drag_id = ''
|
|
|
|
self.bind('<Configure>', self.dragging)
|
2022-04-11 18:35:28 +01:00
|
|
|
|
2023-07-11 08:35:23 +01:00
|
|
|
self.after(1, self.healthcheck_step)
|
2023-06-29 17:15:03 +01:00
|
|
|
|
2023-06-26 13:55:50 +01:00
|
|
|
def __str__(self):
|
2025-01-15 20:56:37 +00:00
|
|
|
return f'{type(self).__name__}App'
|
2023-06-26 13:55:50 +01:00
|
|
|
|
2022-04-11 18:35:28 +01:00
|
|
|
@property
|
|
|
|
def target(self):
|
|
|
|
"""returns the current interface"""
|
|
|
|
|
2022-05-10 20:34:29 +01:00
|
|
|
return self._vban if _base_values.vban_connected else self._vmr
|
2022-04-11 18:35:28 +01:00
|
|
|
|
|
|
|
@property
|
|
|
|
def configframes(self):
|
2022-05-10 20:34:29 +01:00
|
|
|
"""returns the current configframes"""
|
|
|
|
|
|
|
|
return (
|
2022-04-11 18:35:28 +01:00
|
|
|
frame
|
|
|
|
for frame in self.winfo_children()
|
|
|
|
if isinstance(frame, ttk.Frame)
|
2025-01-15 20:56:37 +00:00
|
|
|
and '!stripconfig' in str(frame)
|
|
|
|
or '!busconfig' in str(frame)
|
2022-04-11 18:35:28 +01:00
|
|
|
)
|
|
|
|
|
2022-05-10 20:34:29 +01:00
|
|
|
def build_app(self, kind=None, vban=None):
|
|
|
|
"""builds the app frames according to a kind"""
|
2022-04-11 18:35:28 +01:00
|
|
|
self._vban = vban
|
2022-05-10 20:34:29 +01:00
|
|
|
if kind:
|
|
|
|
self.kind = kind
|
2022-05-14 14:05:48 +01:00
|
|
|
|
2023-06-26 13:56:06 +01:00
|
|
|
# register event callbacks
|
2023-06-26 13:55:50 +01:00
|
|
|
self.target.subject.add([self.on_pdirty, self.on_ldirty])
|
2022-04-11 18:35:28 +01:00
|
|
|
|
|
|
|
self.bus_frame = None
|
2022-05-10 20:34:29 +01:00
|
|
|
self.submix_frame = None
|
|
|
|
self.builder = MainFrameBuilder(self)
|
|
|
|
self.builder.setup()
|
2025-01-15 20:56:37 +00:00
|
|
|
self.builder.create_channelframe('strip')
|
2022-05-10 20:34:29 +01:00
|
|
|
self.builder.create_separator()
|
|
|
|
self.builder.create_navframe()
|
|
|
|
if _configuration.extended:
|
2022-04-11 18:35:28 +01:00
|
|
|
self.nav_frame.extend.set(True)
|
|
|
|
self.nav_frame.extend_frame()
|
2025-01-15 20:56:37 +00:00
|
|
|
if self.kind.name == 'potato':
|
2022-05-10 20:34:29 +01:00
|
|
|
self.builder.create_banner()
|
|
|
|
|
2023-06-26 13:55:50 +01:00
|
|
|
def on_pdirty(self):
|
|
|
|
if _base_values.run_update:
|
2025-01-15 20:56:37 +00:00
|
|
|
self.after(1, self.subject.notify, 'pdirty')
|
2023-06-26 13:55:50 +01:00
|
|
|
|
|
|
|
def on_ldirty(self):
|
|
|
|
if not _base_values.dragging:
|
2025-01-15 20:56:37 +00:00
|
|
|
self.after(1, self.subject.notify, 'ldirty')
|
2022-04-11 18:35:28 +01:00
|
|
|
|
|
|
|
def _destroy_top_level_frames(self):
|
2022-05-10 20:34:29 +01:00
|
|
|
"""
|
|
|
|
Clear observables.
|
|
|
|
|
2022-05-14 14:05:48 +01:00
|
|
|
Deregister app as observer.
|
2022-05-10 20:34:29 +01:00
|
|
|
|
|
|
|
Destroy all top level frames.
|
|
|
|
"""
|
2023-06-30 04:27:10 +01:00
|
|
|
self.target.subject.remove([self.on_pdirty, self.on_ldirty])
|
2022-06-20 00:09:27 +01:00
|
|
|
self.subject.clear()
|
2022-04-11 18:35:28 +01:00
|
|
|
[
|
|
|
|
frame.destroy()
|
|
|
|
for frame in self.winfo_children()
|
|
|
|
if isinstance(frame, ttk.Frame)
|
|
|
|
]
|
|
|
|
|
|
|
|
def dragging(self, event, *args):
|
|
|
|
if event.widget is self:
|
2025-01-15 20:56:37 +00:00
|
|
|
if self.drag_id == '':
|
2022-05-10 20:34:29 +01:00
|
|
|
_base_values.dragging = True
|
2022-04-11 18:35:28 +01:00
|
|
|
else:
|
|
|
|
self.after_cancel(self.drag_id)
|
|
|
|
self.drag_id = self.after(100, self.stop_drag)
|
|
|
|
|
|
|
|
def stop_drag(self):
|
2025-01-15 20:56:37 +00:00
|
|
|
self.drag_id = ''
|
2022-09-16 09:44:47 +01:00
|
|
|
_base_values.dragging = False
|
2022-04-11 18:35:28 +01:00
|
|
|
|
2023-06-26 16:08:58 +01:00
|
|
|
@cached_property
|
|
|
|
def userconfigs(self):
|
2023-07-08 00:22:07 +01:00
|
|
|
self._configs = loader(self.kind.name, self.target)
|
2023-06-26 16:08:58 +01:00
|
|
|
return self._configs
|
|
|
|
|
2023-07-11 08:35:23 +01:00
|
|
|
def start_updates(self):
|
2023-09-07 08:39:20 +01:00
|
|
|
def init():
|
2025-01-15 20:56:37 +00:00
|
|
|
self.logger.debug('updates started')
|
2023-09-07 08:39:20 +01:00
|
|
|
_base_values.run_update = True
|
|
|
|
|
2023-07-11 08:35:23 +01:00
|
|
|
if self._vmr.gui.launched_by_api:
|
2025-01-15 20:56:37 +00:00
|
|
|
self.subject.notify('pdirty')
|
2023-09-07 08:39:20 +01:00
|
|
|
self.after(12000, init)
|
|
|
|
else:
|
|
|
|
init()
|
2023-07-11 08:35:23 +01:00
|
|
|
|
|
|
|
def healthcheck_step(self):
|
|
|
|
if not _base_values.vban_connected:
|
|
|
|
try:
|
2023-07-13 01:32:01 +01:00
|
|
|
self._vmr.version
|
2023-07-11 08:35:23 +01:00
|
|
|
except voicemeeterlib.error.CAPIError:
|
2025-01-15 20:56:37 +00:00
|
|
|
resp = messagebox.askyesno(message='Restart Voicemeeter GUI?')
|
2023-07-11 08:35:23 +01:00
|
|
|
if resp:
|
|
|
|
self.logger.debug(
|
2025-01-15 20:56:37 +00:00
|
|
|
'healthcheck failed, rebuilding the app after GUI restart.'
|
2023-07-11 08:35:23 +01:00
|
|
|
)
|
|
|
|
self._vmr.end_thread()
|
|
|
|
self._vmr.run_voicemeeter(self._vmr.kind.name)
|
|
|
|
_base_values.run_update = False
|
|
|
|
self._vmr.init_thread()
|
|
|
|
self.after(8000, self.start_updates)
|
|
|
|
self._destroy_top_level_frames()
|
|
|
|
self.build_app(self._vmr.kind)
|
2025-01-15 20:56:37 +00:00
|
|
|
vban_config = get_configuration('vban')
|
2023-07-11 08:35:23 +01:00
|
|
|
for i, _ in enumerate(vban_config):
|
2025-01-15 20:56:37 +00:00
|
|
|
target = getattr(self.menu, f'menu_vban_{i + 1}')
|
|
|
|
target.entryconfig(0, state='normal')
|
|
|
|
target.entryconfig(1, state='disabled')
|
2023-07-11 08:35:23 +01:00
|
|
|
[
|
2025-01-15 20:56:37 +00:00
|
|
|
self.menu.menu_vban.entryconfig(j, state='normal')
|
2023-07-11 08:35:23 +01:00
|
|
|
for j, _ in enumerate(self.menu.menu_vban.winfo_children())
|
|
|
|
]
|
|
|
|
else:
|
|
|
|
self.destroy()
|
|
|
|
self.after(250, self.healthcheck_step)
|
|
|
|
|
2023-08-06 23:15:54 +01:00
|
|
|
def on_close_window(self):
|
|
|
|
if _base_values.vban_connected:
|
|
|
|
self._vban.logout()
|
|
|
|
self.destroy()
|
|
|
|
|
2022-04-11 18:35:28 +01:00
|
|
|
|
2022-06-16 23:53:28 +01:00
|
|
|
_apps = {kind.name: App.make(kind) for kind in _kinds_all}
|
2022-04-11 18:35:28 +01:00
|
|
|
|
|
|
|
|
|
|
|
def connect(kind_id: str, vmr) -> App:
|
|
|
|
"""return App of the kind requested"""
|
2022-05-10 20:34:29 +01:00
|
|
|
|
2022-04-11 18:35:28 +01:00
|
|
|
try:
|
|
|
|
VMMIN_cls = _apps[kind_id]
|
|
|
|
except KeyError:
|
2025-01-15 20:56:37 +00:00
|
|
|
raise VMCompactError(f'Invalid kind: {kind_id}')
|
2022-08-02 10:06:35 +01:00
|
|
|
return VMMIN_cls(vmr)
|