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>
207 lines
7.3 KiB
Python
207 lines
7.3 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 dataclasses
|
|
import functools
|
|
import unittest
|
|
import itertools
|
|
from typing import Iterable, List
|
|
|
|
from inputremapper.injection.mapping_handlers.axis_transform import Transformation
|
|
|
|
|
|
class TestAxisTransformation(unittest.TestCase):
|
|
@dataclasses.dataclass
|
|
class InitArgs:
|
|
max_: int
|
|
min_: int
|
|
deadzone: float
|
|
gain: float
|
|
expo: float
|
|
|
|
def values(self):
|
|
return self.__dict__.values()
|
|
|
|
def get_init_args(
|
|
self,
|
|
max_=(255, 1000, 2**15),
|
|
min_=(50, 0, -255),
|
|
deadzone=(0, 0.5),
|
|
gain=(0.5, 1, 2),
|
|
expo=(-0.9, 0, 0.3),
|
|
) -> Iterable[InitArgs]:
|
|
for args in itertools.product(max_, min_, deadzone, gain, expo):
|
|
yield self.InitArgs(*args)
|
|
|
|
@staticmethod
|
|
def scale_to_range(min_, max_, x=(-1, -0.2, 0, 0.6, 1)) -> List[float]:
|
|
"""Scale values between -1 and 1 up, such that they are between min and max."""
|
|
half_range = (max_ - min_) / 2
|
|
return [float_x * half_range + min_ + half_range for float_x in x]
|
|
|
|
def test_scale_to_range(self):
|
|
"""Make sure scale_to_range will actually return the min and max values
|
|
(avoid "off by one" errors)"""
|
|
max_ = (255, 1000, 2**15)
|
|
min_ = (50, 0, -255)
|
|
|
|
for x1, x2 in itertools.product(min_, max_):
|
|
scaled = self.scale_to_range(x1, x2, (-1, 1))
|
|
self.assertEqual(scaled, [x1, x2])
|
|
|
|
def test_expo_symmetry(self):
|
|
"""Test that the transformation is symmetric for expo parameter
|
|
x = f(g(x)), if f._expo == - g._expo
|
|
|
|
with the following constraints:
|
|
min = -1, max = 1
|
|
gain = 1
|
|
deadzone = 0
|
|
|
|
we can remove the constraints for min, max and gain,
|
|
by scaling the values appropriately after each transformation
|
|
"""
|
|
|
|
for init_args in self.get_init_args(deadzone=(0,)):
|
|
f = Transformation(*init_args.values())
|
|
init_args.expo = -init_args.expo
|
|
g = Transformation(*init_args.values())
|
|
|
|
scale = functools.partial(
|
|
self.scale_to_range,
|
|
init_args.min_,
|
|
init_args.max_,
|
|
)
|
|
for x in scale():
|
|
y1 = g(x)
|
|
y1 = y1 / init_args.gain # remove the gain
|
|
y1 = scale((y1,))[0] # remove the min/max constraint
|
|
|
|
y2 = f(y1)
|
|
y2 = y2 / init_args.gain # remove the gain
|
|
y2 = scale((y2,))[0] # remove the min/max constraint
|
|
self.assertAlmostEqual(x, y2, msg=f"test expo symmetry for {init_args}")
|
|
|
|
def test_origin_symmetry(self):
|
|
"""Test that the transformation is symmetric to the origin
|
|
f(x) = - f(-x)
|
|
within the constraints: min = -max
|
|
"""
|
|
|
|
for init_args in self.get_init_args():
|
|
init_args.min_ = -init_args.max_
|
|
f = Transformation(*init_args.values())
|
|
for x in self.scale_to_range(init_args.min_, init_args.max_):
|
|
self.assertAlmostEqual(
|
|
f(x),
|
|
-f(-x),
|
|
msg=f"test origin symmetry at {x=} for {init_args}",
|
|
)
|
|
|
|
def test_gain(self):
|
|
"""Test that f(max) = gain and f(min) = -gain."""
|
|
for init_args in self.get_init_args():
|
|
f = Transformation(*init_args.values())
|
|
self.assertAlmostEqual(
|
|
f(init_args.max_),
|
|
init_args.gain,
|
|
msg=f"test gain for {init_args}",
|
|
)
|
|
self.assertAlmostEqual(
|
|
f(init_args.min_),
|
|
-init_args.gain,
|
|
msg=f"test gain for {init_args}",
|
|
)
|
|
|
|
def test_deadzone(self):
|
|
"""Test the Transfomation returns exactly 0 in the range of the deadzone."""
|
|
|
|
for init_args in self.get_init_args(deadzone=(0.1, 0.2, 0.9)):
|
|
f = Transformation(*init_args.values())
|
|
for x in self.scale_to_range(
|
|
init_args.min_,
|
|
init_args.max_,
|
|
x=(
|
|
init_args.deadzone * 0.999,
|
|
-init_args.deadzone * 0.999,
|
|
0.3 * init_args.deadzone,
|
|
0,
|
|
),
|
|
):
|
|
self.assertEqual(f(x), 0, msg=f"test deadzone at {x=} for {init_args}")
|
|
|
|
def test_continuity_near_deadzone(self):
|
|
"""Test that the Transfomation is continues (no sudden jump) next to the
|
|
deadzone"""
|
|
|
|
for init_args in self.get_init_args(deadzone=(0.1, 0.2, 0.9)):
|
|
f = Transformation(*init_args.values())
|
|
scale = functools.partial(
|
|
self.scale_to_range,
|
|
init_args.min_,
|
|
init_args.max_,
|
|
)
|
|
x = (
|
|
init_args.deadzone * 1.00001,
|
|
init_args.deadzone * 1.001,
|
|
-init_args.deadzone * 1.00001,
|
|
-init_args.deadzone * 1.001,
|
|
)
|
|
scaled_x = scale(x=x)
|
|
|
|
p1 = (x[0], f(scaled_x[0])) # first point right of deadzone
|
|
p2 = (x[1], f(scaled_x[1])) # second point right of deadzone
|
|
|
|
# calculate a linear function y = m * x + b from p1 and p2
|
|
m = (p1[1] - p2[1]) / (p1[0] - p2[0])
|
|
b = p1[1] - m * p1[0]
|
|
|
|
# the zero intersection of that function must be close to the
|
|
# edge of the deadzone
|
|
self.assertAlmostEqual(
|
|
-b / m,
|
|
init_args.deadzone,
|
|
places=5,
|
|
msg=f"test continuity at {init_args.deadzone} for {init_args}",
|
|
)
|
|
|
|
# same thing on the other side
|
|
p1 = (x[2], f(scaled_x[2]))
|
|
p2 = (x[3], f(scaled_x[3]))
|
|
m = (p1[1] - p2[1]) / (p1[0] - p2[0])
|
|
b = p1[1] - m * p1[0]
|
|
self.assertAlmostEqual(
|
|
-b / m,
|
|
-init_args.deadzone,
|
|
places=5,
|
|
msg=f"test continuity at {- init_args.deadzone} for {init_args}",
|
|
)
|
|
|
|
def test_expo_out_of_range(self):
|
|
f = Transformation(deadzone=0.1, min_=-20, max_=5, expo=1.3)
|
|
self.assertRaises(ValueError, f, 0)
|
|
f = Transformation(deadzone=0.1, min_=-20, max_=5, expo=-1.3)
|
|
self.assertRaises(ValueError, f, 0)
|
|
|
|
def test_returns_one_for_range_between_minus_and_plus_one(self):
|
|
for init_args in self.get_init_args(max_=(1,), min_=(-1,), gain=(1,)):
|
|
f = Transformation(*init_args.values())
|
|
self.assertEqual(f(1), 1)
|
|
self.assertEqual(f(-1), -1)
|