mirror of
https://github.com/sezanzeb/input-remapper
synced 2024-11-12 01:10:38 +00:00
3637204bff
* Tests for the GuiEventHandler * Implement GuiEventHandler * tests for data manager * Implemented data_manager * Remove Ellipsis from type hint * workaround for old pydantic version * workaround for old pydantic version * some more tests for data_manager * Updated Data Manager * move DeviceSelection to its own class * Data Manager no longer listens for events * Moved PresetSelection to its own class * MappingListBox and SelectionLable Listen to the EventHandler * DataManager no longer creates its own data objects in the init * removed global reader object * Changed UI startup * created backend Interface * event_handler debug logs show function which emit a event * some cleanup * added target selector to components * created code editor component * adapted autocompletion & some cleanup * black * connected some buttons to the event_handler * tests for data_manager newest_preset and group * cleanup presets and test_presets * migrated confirm delete dialog * backend tests * controller tests * add python3-gi to ci * more dependencies * and more ... * Github-Actions workaround remove this commit * not so many permission denyed errors in test.yml * Fix #404 (hopefully) * revert Github-Actions workaround * More tests * event_handler allows for event supression * more tests * WIP Implement Key recording * Start and Stop Injection * context no longer stores preset * restructured the RelToBtnHandler * Simplified read_loop * Implement async iterator for ipc.pipe * multiple event actions * helper now implements mapping handlers to read inputs all with async * updated and simplified reader the helper uses the mapping handlers, so the reader now can be much simpler * Fixed race condition in tests * implemented DataBus * Fixed a UIMapping bug where the last_error would not be deleted * added a immutable variant of the UIMapping * updated data_manager to use data_bus * Uptdated tests to use the DataBus * Gui uses DataBus * removed EventHandler * Renamed controller methods * Implemented recording toggle * implemented StatusBar * Sending validation errors to status bar * sending injection status to status bar * proper preset renaming * implemented copy preset in the data manager * implemented copy_preset in controller * fixed a bug where a wron selection lable would update * no longer send invalid data over the bus, if the preset or group changes * Implement create and delete mapping * Allow for frontend specific mapping defaults * implemented autoload toggle * cleanup user_interface * removed editor * Docstings renaming and ordering of methods * more simplifications to user_interface * integrated backend into data_manager * removed active preset * transformation tests * controller tests * fix missing uinputs in gui * moved some tests and implemented basic tests for mapping handlers * docstring reformatting Co-authored-by: Tobi <proxima@sezanzeb.de> * allow for empty groups * docstring * fixed TestGroupFromHelper * some work on integration tests * test for annoying import error in tests * testing if test_user_interface works * I feel lucky * not so lucky * some more tests * fixed but where the group_key was used as folder name * Fixed a bug where state=NO_GRAB would never be read from the injector * allow to stop the recorder * working on integration tests * integration tests * fixed more integration tests * updated coveragerc * no longer attempt to record keys when injecting * event_reader cleans up not finished tasks * More integration tests * All tests pass * renamed data_bus * WIP fixing typing issues * more typing fixes * added keyboard+mouse device to tests * cleanup imports * new read loop because the evdev async read loop can not be cancelled * Added field to modify mapping name * created tests for components * even more component tests * do component tests need a screen? * apparently they do :_( * created release_input switch * Don't record relative axis when movement is slow * show delete dialog above main window * wip basic dialog to edit combination * some gui changes to the combination-editor * Simple implementation of CombinationListbox * renamed attach_to_events method and mark as private * shorter str() for UInputsData * moved logic to generate readable event string from combination to event * new mapping parameter force release timeout this helps with the helper when recording multiple relative axis at once * make it possible to rearange the event_combination * more work on the combination editor * tests for DataManager.load_event * simplyfied test_controller * more controller tests * Implement input threshold in gui * greater range for time dependent unit test * implemented a output-axis selector * data_manager now provides injector state * black * mypy * Updated confirm cancel dialog * created release timeout input * implemented transformation graph * Added sliders for gain, expo and deadzone * fix bug where the system_mapping was overridden in each injector thread * updated slider settings * removed debug statement * explicitly checking output code against None (0 is a valid code) * usage * Allow for multiple axis to be activated by same button * readme * only warn about not implemented mapping-handler don't fail to create event-pipelines * More accurate event names * Allow removal of single events from the input-combination * rename callback to notify_callback * rename event message to selected_event * made read_continuisly private * typing for autocompletion * docstrings for message_broker messages * make components methods and propreties private * gui spacings * removed eval * make some controller functions private * move status message generation from data_manager to controller * parse mapping errors in controller for more helpful messages * remove system_mapping from code editor * More component tests * more tests * mypy * make grab_devices less greedy (partial mitigation for #435) only grab one device if there are multiple which can satisfy the same mapping * accumulate more values in test * docstrings * Updated status messages * comments, docstrings, imports Co-authored-by: Tobi <proxima@sezanzeb.de>
239 lines
6.7 KiB
Python
239 lines
6.7 KiB
Python
#!/usr/bin/python3
|
|
# -*- coding: utf-8 -*-
|
|
# input-remapper - GUI for device specific keyboard mappings
|
|
# Copyright (C) 2022 sezanzeb <proxima@sezanzeb.de>
|
|
#
|
|
# This file is part of input-remapper.
|
|
#
|
|
# input-remapper is free software: you can redistribute it and/or modify
|
|
# it under the terms of the GNU General Public License as published by
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
# (at your option) any later version.
|
|
#
|
|
# input-remapper is distributed in the hope that it will be useful,
|
|
# but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
# GNU General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU General Public License
|
|
# along with input-remapper. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
import os.path
|
|
import re
|
|
import traceback
|
|
from collections import defaultdict, deque
|
|
from dataclasses import dataclass
|
|
from enum import Enum
|
|
from typing import (
|
|
Callable,
|
|
Dict,
|
|
Set,
|
|
Protocol,
|
|
Tuple,
|
|
Deque,
|
|
Optional,
|
|
List,
|
|
Any,
|
|
TYPE_CHECKING,
|
|
)
|
|
|
|
from inputremapper.groups import DeviceType
|
|
from inputremapper.logger import logger
|
|
|
|
if TYPE_CHECKING:
|
|
from inputremapper.event_combination import EventCombination
|
|
|
|
|
|
class MessageType(Enum):
|
|
reset_gui = "reset_gui"
|
|
terminate = "terminate"
|
|
init = "init"
|
|
|
|
uinputs = "uinputs"
|
|
groups = "groups"
|
|
group = "group"
|
|
preset = "preset"
|
|
mapping = "mapping"
|
|
selected_event = "selected_event"
|
|
combination_recorded = "combination_recorded"
|
|
recording_finished = "recording_finished"
|
|
combination_update = "combination_update"
|
|
status_msg = "status_msg"
|
|
injector_state = "injector_state"
|
|
|
|
gui_focus_request = "gui_focus_request"
|
|
user_confirm_request = "user_confirm_request"
|
|
|
|
# for unit tests:
|
|
test1 = "test1"
|
|
test2 = "test2"
|
|
|
|
|
|
class Message(Protocol):
|
|
"""the protocol any message must follow to be sent with the MessageBroker"""
|
|
|
|
message_type: MessageType
|
|
|
|
|
|
# useful type aliases
|
|
MessageListener = Callable[[Any], None]
|
|
Capabilities = Dict[int, List]
|
|
Name = str
|
|
Key = str
|
|
DeviceTypes = List[DeviceType]
|
|
|
|
|
|
class MessageBroker:
|
|
shorten_path = re.compile("inputremapper/")
|
|
|
|
def __init__(self):
|
|
self._listeners: Dict[MessageType, Set[MessageListener]] = defaultdict(set)
|
|
self._messages: Deque[Tuple[Message, str, int]] = deque()
|
|
self._sending = False
|
|
|
|
def send(self, data: Message):
|
|
"""schedule a massage to be sent.
|
|
The message will be sent after all currently pending messages are sent"""
|
|
self._messages.append((data, *self.get_caller()))
|
|
self._send_all()
|
|
|
|
def signal(self, signal: MessageType):
|
|
"""send a signal without any data payload"""
|
|
self.send(Signal(signal))
|
|
|
|
def _send(self, data: Message, file: str, line: int):
|
|
logger.debug(f"from {file}:{line}: Signal={data.message_type.name}: {data}")
|
|
for listener in self._listeners[data.message_type].copy():
|
|
listener(data)
|
|
|
|
def _send_all(self):
|
|
"""send all scheduled messages in order"""
|
|
if self._sending:
|
|
# don't run this twice, so we not mess up the order
|
|
return
|
|
|
|
self._sending = True
|
|
try:
|
|
while self._messages:
|
|
self._send(*self._messages.popleft())
|
|
finally:
|
|
self._sending = False
|
|
|
|
def subscribe(self, massage_type: MessageType, listener: MessageListener):
|
|
"""attach a listener to an event"""
|
|
logger.debug("adding new Listener: %s", listener)
|
|
self._listeners[massage_type].add(listener)
|
|
return self
|
|
|
|
@staticmethod
|
|
def get_caller(position: int = 3) -> Tuple[str, int]:
|
|
"""extract a file and line from current stack and format for logging"""
|
|
tb = traceback.extract_stack(limit=position)[0]
|
|
return os.path.basename(tb.filename), tb.lineno or 0
|
|
|
|
def unsubscribe(self, listener: MessageListener) -> None:
|
|
for listeners in self._listeners.values():
|
|
try:
|
|
listeners.remove(listener)
|
|
except KeyError:
|
|
pass
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class UInputsData:
|
|
message_type = MessageType.uinputs
|
|
uinputs: Dict[Name, Capabilities]
|
|
|
|
def __str__(self):
|
|
string = f"{self.__class__.__name__}(uinputs={self.uinputs})"
|
|
|
|
# find all sequences of comma+space separated numbers, and shorten them
|
|
# to the first and last number
|
|
all_matches = [m for m in re.finditer("(\d+, )+", string)]
|
|
all_matches.reverse()
|
|
for match in all_matches:
|
|
start = match.start()
|
|
end = match.end()
|
|
start += string[start:].find(",") + 2
|
|
if start == end:
|
|
continue
|
|
string = f"{string[:start]}... {string[end:]}"
|
|
|
|
return string
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class GroupsData:
|
|
"""Message containing all available groups and their device types"""
|
|
|
|
message_type = MessageType.groups
|
|
groups: Dict[Key, DeviceTypes]
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class GroupData:
|
|
"""Message with the active group and available presets for the group"""
|
|
|
|
message_type = MessageType.group
|
|
group_key: str
|
|
presets: Tuple[str, ...]
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class PresetData:
|
|
"""Message with the active preset name and mapping names/combinations"""
|
|
|
|
message_type = MessageType.preset
|
|
name: Optional[Name]
|
|
mappings: Optional[Tuple[Tuple[Name, "EventCombination"], ...]]
|
|
autoload: bool = False
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class StatusData:
|
|
"""Message with the strings and id for the status bar"""
|
|
|
|
message_type = MessageType.status_msg
|
|
ctx_id: int
|
|
msg: Optional[str] = None
|
|
tooltip: Optional[str] = None
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class CombinationRecorded:
|
|
"""Message with the latest recoded combination"""
|
|
|
|
message_type = MessageType.combination_recorded
|
|
combination: "EventCombination"
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class CombinationUpdate:
|
|
"""Message with the old and new combination (hash for a mapping) when it changed"""
|
|
|
|
message_type = MessageType.combination_update
|
|
old_combination: "EventCombination"
|
|
new_combination: "EventCombination"
|
|
|
|
|
|
@dataclass(frozen=True)
|
|
class UserConfirmRequest:
|
|
"""Message for requesting a user response (confirm/cancel) from the gui"""
|
|
|
|
message_type = MessageType.user_confirm_request
|
|
msg: str
|
|
respond: Callable[[bool], None] = lambda _: None
|
|
|
|
|
|
class Signal(Message):
|
|
"""Send a Message without any associated data over the MassageBus"""
|
|
|
|
def __init__(self, message_type: MessageType):
|
|
self.message_type: MessageType = message_type
|
|
|
|
def __str__(self):
|
|
return f"Signal: {self.message_type}"
|
|
|
|
def __eq__(self, other):
|
|
return str(self) == str(other)
|