input-remapper/tests/integration/test_components.py

1148 lines
42 KiB
Python
Raw Normal View History

Frontend Refactor (#375) * 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>
2022-07-23 08:53:41 +00:00
import unittest
from typing import Optional, Tuple
from unittest.mock import MagicMock, patch
2022-07-25 08:55:26 +00:00
import evdev
Frontend Refactor (#375) * 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>
2022-07-23 08:53:41 +00:00
from evdev.ecodes import EV_KEY, KEY_A, KEY_B, KEY_C
import gi
from inputremapper.input_event import InputEvent
gi.require_version("Gtk", "3.0")
gi.require_version("GLib", "2.0")
gi.require_version("GtkSource", "4")
from gi.repository import Gtk, GLib, GtkSource
from tests.test import quick_cleanup, spy
from inputremapper.gui.utils import CTX_ERROR, CTX_WARNING, gtk_iteration
from inputremapper.gui.message_broker import (
MessageBroker,
MessageType,
GroupData,
GroupsData,
UInputsData,
PresetData,
CombinationUpdate,
StatusData,
)
from inputremapper.groups import DeviceType
from inputremapper.gui.components import (
DeviceSelection,
TargetSelection,
PresetSelection,
MappingListBox,
SelectionLabel,
CodeEditor,
RecordingToggle,
StatusBar,
AutoloadSwitch,
ReleaseCombinationSwitch,
CombinationListbox,
EventEntry,
AnalogInputSwitch,
TriggerThresholdInput,
ReleaseTimeoutInput,
OutputAxisSelector,
KeyAxisStack,
Sliders,
TransformationDrawArea,
)
from inputremapper.configs.mapping import MappingData
from inputremapper.event_combination import EventCombination
class ComponentBaseTest(unittest.TestCase):
"""test a gui component
ensures to tearDown self.gui
all gtk objects must be a child of self.gui in order to ensure proper cleanup"""
def setUp(self) -> None:
self.message_broker = MessageBroker()
self.controller_mock = MagicMock()
self.gui = MagicMock()
def tearDown(self) -> None:
super().tearDown()
self.message_broker.signal(MessageType.terminate)
GLib.timeout_add(0, self.gui.destroy)
GLib.timeout_add(0, Gtk.main_quit)
Gtk.main()
quick_cleanup()
class TestDeviceSelection(ComponentBaseTest):
def setUp(self) -> None:
super(TestDeviceSelection, self).setUp()
self.gui = Gtk.ComboBox()
self.selection = DeviceSelection(
self.message_broker, self.controller_mock, self.gui
)
self.message_broker.send(
GroupsData(
{
"foo": [DeviceType.GAMEPAD, DeviceType.KEYBOARD],
"bar": [],
"baz": [DeviceType.GRAPHICS_TABLET],
}
)
)
def test_populates_devices(self):
names = [row[0] for row in self.gui.get_model()]
self.assertEqual(names, ["foo", "bar", "baz"])
icons = [row[1] for row in self.gui.get_model()]
self.assertEqual(icons, ["input-gaming", None, "input-tablet"])
self.message_broker.send(
GroupsData(
{
"kuu": [DeviceType.KEYBOARD],
"qux": [DeviceType.GAMEPAD],
}
)
)
names = [row[0] for row in self.gui.get_model()]
self.assertEqual(names, ["kuu", "qux"])
icons = [row[1] for row in self.gui.get_model()]
self.assertEqual(icons, ["input-keyboard", "input-gaming"])
def test_selects_correct_device(self):
self.message_broker.send(GroupData("bar", ()))
self.assertEqual(self.gui.get_active_id(), "bar")
self.message_broker.send(GroupData("baz", ()))
self.assertEqual(self.gui.get_active_id(), "baz")
def test_loads_group(self):
self.gui.set_active_id("bar")
self.controller_mock.load_group.assert_called_once_with("bar")
def test_avoids_infinite_recursion(self):
self.message_broker.send(GroupData("bar", ()))
self.controller_mock.load_group.assert_not_called()
class TestTargetSelection(ComponentBaseTest):
def setUp(self) -> None:
super(TestTargetSelection, self).setUp()
self.gui = Gtk.ComboBox()
self.selection = TargetSelection(
self.message_broker, self.controller_mock, self.gui
)
self.message_broker.send(
UInputsData(
{
"foo": {},
"bar": {},
"baz": {},
}
)
)
def test_populates_devices(self):
names = [row[0] for row in self.gui.get_model()]
self.assertEqual(names, ["foo", "bar", "baz"])
self.message_broker.send(
UInputsData(
{
"kuu": {},
"qux": {},
}
)
)
names = [row[0] for row in self.gui.get_model()]
self.assertEqual(names, ["kuu", "qux"])
def test_updates_mapping(self):
self.gui.set_active_id("baz")
self.controller_mock.update_mapping.called_once_with(target_uinput="baz")
def test_selects_correct_target(self):
self.message_broker.send(MappingData(target_uinput="baz"))
self.assertEqual(self.gui.get_active_id(), "baz")
self.message_broker.send(MappingData(target_uinput="bar"))
self.assertEqual(self.gui.get_active_id(), "bar")
def test_avoids_infinite_recursion(self):
self.message_broker.send(MappingData(target_uinput="baz"))
self.controller_mock.update_mapping.assert_not_called()
def test_disabled_with_invalid_mapping(self):
self.controller_mock.is_empty_mapping.return_value = True
self.message_broker.send(MappingData())
self.assertFalse(self.gui.get_sensitive())
self.assertLess(self.gui.get_opacity(), 0.8)
def test_enabled_with_valid_mapping(self):
self.controller_mock.is_empty_mapping.return_value = False
self.message_broker.send(MappingData())
self.assertTrue(self.gui.get_sensitive())
self.assertEqual(self.gui.get_opacity(), 1)
class TestPresetSelection(ComponentBaseTest):
def setUp(self) -> None:
super().setUp()
self.gui = Gtk.ComboBoxText()
self.selection = PresetSelection(
self.message_broker, self.controller_mock, self.gui
)
self.message_broker.send(GroupData("foo", ("preset1", "preset2")))
def test_populates_presets(self):
names = [row[0] for row in self.gui.get_model()]
self.assertEqual(names, ["preset1", "preset2"])
self.message_broker.send(GroupData("foo", ("preset3", "preset4")))
names = [row[0] for row in self.gui.get_model()]
self.assertEqual(names, ["preset3", "preset4"])
def test_selects_preset(self):
self.message_broker.send(
PresetData("preset2", (("m1", EventCombination((1, 2, 3))),))
)
self.assertEqual(self.gui.get_active_id(), "preset2")
self.message_broker.send(
PresetData("preset1", (("m1", EventCombination((1, 2, 3))),))
)
self.assertEqual(self.gui.get_active_id(), "preset1")
def test_avoids_infinite_recursion(self):
self.message_broker.send(
PresetData("preset2", (("m1", EventCombination((1, 2, 3))),))
)
self.controller_mock.load_preset.assert_not_called()
def test_loads_preset(self):
self.gui.set_active_id("preset2")
self.controller_mock.load_preset.assert_called_once_with("preset2")
class TestMappingListbox(ComponentBaseTest):
def setUp(self) -> None:
super().setUp()
self.gui = Gtk.ListBox()
self.listbox = MappingListBox(
self.message_broker, self.controller_mock, self.gui
)
self.message_broker.send(
PresetData(
"preset1",
(
("mapping1", EventCombination((1, KEY_C, 1))),
("", EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)])),
("mapping2", EventCombination((1, KEY_B, 1))),
),
)
)
def get_selected_row(self) -> SelectionLabel:
row = None
def find_row(r: SelectionLabel):
nonlocal row
if r.is_selected():
row = r
self.gui.foreach(find_row)
assert row is not None
return row
def select_row(self, combination: EventCombination):
def select(row: SelectionLabel):
if row.combination == combination:
self.gui.select_row(row)
self.gui.foreach(select)
def test_populates_listbox(self):
labels = {row.name for row in self.gui.get_children()}
self.assertEqual(labels, {"mapping1", "mapping2", "a + b"})
def test_alphanumerically_sorted(self):
labels = [row.name for row in self.gui.get_children()]
self.assertEqual(labels, ["a + b", "mapping1", "mapping2"])
def test_activates_correct_row(self):
self.message_broker.send(
MappingData(
name="mapping1", event_combination=EventCombination((1, KEY_C, 1))
)
)
selected = self.get_selected_row()
self.assertEqual(selected.name, "mapping1")
self.assertEqual(selected.combination, EventCombination((1, KEY_C, 1)))
def test_loads_mapping(self):
self.select_row(EventCombination((1, KEY_B, 1)))
self.controller_mock.load_mapping.assert_called_once_with(
EventCombination((1, KEY_B, 1))
)
def test_avoids_infinite_recursion(self):
self.message_broker.send(
MappingData(
name="mapping1", event_combination=EventCombination((1, KEY_C, 1))
)
)
self.controller_mock.load_mapping.assert_not_called()
def test_sorts_empty_mapping_to_bottom(self):
self.message_broker.send(
PresetData(
"preset1",
(
("qux", EventCombination((1, KEY_C, 1))),
("foo", EventCombination.empty_combination()),
("bar", EventCombination((1, KEY_B, 1))),
),
)
)
bottom_row: SelectionLabel = self.gui.get_row_at_index(2)
self.assertEqual(bottom_row.combination, EventCombination.empty_combination())
self.message_broker.send(
PresetData(
"preset1",
(
("foo", EventCombination.empty_combination()),
("qux", EventCombination((1, KEY_C, 1))),
("bar", EventCombination((1, KEY_B, 1))),
),
)
)
bottom_row: SelectionLabel = self.gui.get_row_at_index(2)
self.assertEqual(bottom_row.combination, EventCombination.empty_combination())
class TestSelectionLabel(ComponentBaseTest):
def setUp(self) -> None:
super().setUp()
self.gui = Gtk.ListBox()
self.label = SelectionLabel(
self.message_broker,
self.controller_mock,
"",
EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)]),
)
self.gui.insert(self.label, -1)
def test_shows_combination_without_name(self):
self.assertEqual(self.label.label.get_label(), "a + b")
def test_shows_name_when_given(self):
self.gui = SelectionLabel(
self.message_broker,
self.controller_mock,
"foo",
EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)]),
)
self.assertEqual(self.gui.label.get_label(), "foo")
def test_updates_combination_when_selected(self):
self.gui.select_row(self.label)
self.assertEqual(
self.label.combination, EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)])
)
self.message_broker.send(
CombinationUpdate(
EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)]),
EventCombination((1, KEY_A, 1)),
)
)
self.assertEqual(self.label.combination, EventCombination((1, KEY_A, 1)))
def test_doesnt_update_combination_when_not_selected(self):
self.assertEqual(
self.label.combination, EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)])
)
self.message_broker.send(
CombinationUpdate(
EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)]),
EventCombination((1, KEY_A, 1)),
)
)
self.assertEqual(
self.label.combination, EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)])
)
def test_updates_name_when_mapping_changed_and_combination_matches(self):
self.message_broker.send(
MappingData(
event_combination=EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)]),
name="foo",
)
)
self.assertEqual(self.label.label.get_label(), "foo")
def test_ignores_mapping_when_combination_does_not_match(self):
self.message_broker.send(
MappingData(
event_combination=EventCombination([(1, KEY_A, 1), (1, KEY_C, 1)]),
name="foo",
)
)
self.assertEqual(self.label.label.get_label(), "a + b")
def test_edit_button_visibility(self):
# start off invisible
self.assertFalse(self.label.edit_btn.get_visible())
# load the mapping associated with the ListBoxRow
self.message_broker.send(
MappingData(
event_combination=EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)]),
)
)
self.assertTrue(self.label.edit_btn.get_visible())
# load a different row
self.message_broker.send(
MappingData(
event_combination=EventCombination([(1, KEY_A, 1), (1, KEY_C, 1)]),
)
)
self.assertFalse(self.label.edit_btn.get_visible())
def test_enter_edit_mode_focuses_name_input(self):
self.message_broker.send(
MappingData(
event_combination=EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)]),
)
)
self.label.edit_btn.clicked()
self.controller_mock.set_focus.assert_called_once_with(self.label.name_input)
def test_enter_edit_mode_updates_visibility(self):
self.message_broker.send(
MappingData(
event_combination=EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)]),
)
)
self.assertTrue(self.label.label.get_visible())
self.assertFalse(self.label.name_input.get_visible())
self.label.edit_btn.clicked()
self.assertTrue(self.label.name_input.get_visible())
self.assertFalse(self.label.label.get_visible())
self.label.name_input.activate() # aka hit the return key
self.assertTrue(self.label.label.get_visible())
self.assertFalse(self.label.name_input.get_visible())
def test_update_name(self):
self.message_broker.send(
MappingData(
event_combination=EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)]),
)
)
self.label.edit_btn.clicked()
self.label.name_input.set_text("foo")
self.label.name_input.activate()
self.controller_mock.update_mapping.assert_called_once_with(name="foo")
def test_name_input_contains_combination_when_name_not_set(self):
self.message_broker.send(
MappingData(
event_combination=EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)]),
)
)
self.label.edit_btn.clicked()
self.assertEqual(self.label.name_input.get_text(), "a + b")
def test_name_input_contains_name(self):
self.message_broker.send(
MappingData(
event_combination=EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)]),
name="foo",
)
)
self.label.edit_btn.clicked()
self.assertEqual(self.label.name_input.get_text(), "foo")
def test_removes_name_when_name_matches_combination(self):
self.message_broker.send(
MappingData(
event_combination=EventCombination([(1, KEY_A, 1), (1, KEY_B, 1)]),
name="foo",
)
)
self.label.edit_btn.clicked()
self.label.name_input.set_text("a + b")
self.label.name_input.activate()
self.controller_mock.update_mapping.assert_called_once_with(name="")
class TestCodeEditor(ComponentBaseTest):
def setUp(self) -> None:
super(TestCodeEditor, self).setUp()
self.gui = GtkSource.View()
self.editor = CodeEditor(self.message_broker, self.controller_mock, self.gui)
self.controller_mock.is_empty_mapping.return_value = False
def get_text(self) -> str:
buffer = self.gui.get_buffer()
return buffer.get_text(buffer.get_start_iter(), buffer.get_end_iter(), True)
def test_shows_output_symbol(self):
self.message_broker.send(MappingData(output_symbol="foo"))
self.assertEqual(self.get_text(), "foo")
def test_shows_record_input_first_message_when_mapping_is_empty(self):
self.controller_mock.is_empty_mapping.return_value = True
self.message_broker.send(MappingData(output_symbol="foo"))
self.assertEqual(self.get_text(), "Record the input first")
def test_inactive_when_mapping_is_empty(self):
self.controller_mock.is_empty_mapping.return_value = True
self.message_broker.send(MappingData(output_symbol="foo"))
self.assertFalse(self.gui.get_sensitive())
self.assertLess(self.gui.get_opacity(), 0.6)
def test_active_when_mapping_is_not_empty(self):
self.message_broker.send(MappingData(output_symbol="foo"))
self.assertTrue(self.gui.get_sensitive())
self.assertEqual(self.gui.get_opacity(), 1)
def test_expands_to_multiline(self):
self.message_broker.send(MappingData(output_symbol="foo\nbar"))
self.assertIn("multiline", self.gui.get_style_context().list_classes())
def test_shows_line_numbers_when_multiline(self):
self.message_broker.send(MappingData(output_symbol="foo\nbar"))
self.assertTrue(self.gui.get_show_line_numbers())
def test_no_multiline_when_macro_not_multiline(self):
self.message_broker.send(MappingData(output_symbol="foo"))
self.assertNotIn("multiline", self.gui.get_style_context().list_classes())
def test_no_line_numbers_macro_not_multiline(self):
self.message_broker.send(MappingData(output_symbol="foo"))
self.assertFalse(self.gui.get_show_line_numbers())
def test_is_empty_when_mapping_has_no_output_symbol(self):
self.message_broker.send(MappingData())
self.assertEqual(self.get_text(), "")
def test_updates_mapping(self):
self.message_broker.send(MappingData())
buffer = self.gui.get_buffer()
buffer.set_text("foo")
self.controller_mock.update_mapping.assert_called_once_with(output_symbol="foo")
def test_avoids_infinite_recursion_when_loading_mapping(self):
self.message_broker.send(MappingData(output_symbol="foo"))
self.controller_mock.update_mapping.assert_not_called()
def test_gets_focus_when_input_recording_finises(self):
self.message_broker.signal(MessageType.recording_finished)
self.controller_mock.set_focus.assert_called_once_with(self.gui)
class TestRecordingToggle(ComponentBaseTest):
def setUp(self) -> None:
super(TestRecordingToggle, self).setUp()
self.gui = Gtk.ToggleButton()
self.toggle = RecordingToggle(
self.message_broker, self.controller_mock, self.gui
)
def assert_recording(self):
self.assertEqual(self.gui.get_label(), "Recording ...")
self.assertTrue(self.gui.get_active())
def assert_not_recording(self):
self.assertEqual(self.gui.get_label(), "Record Input")
self.assertFalse(self.gui.get_active())
def test_starts_recording(self):
self.gui.set_active(True)
self.controller_mock.start_key_recording.assert_called_once()
def test_stops_recording_when_clicked(self):
self.gui.set_active(True)
self.gui.set_active(False)
self.controller_mock.stop_key_recording.assert_called_once()
def test_not_recording_initially(self):
self.assert_not_recording()
def test_shows_recording_when_toggled(self):
self.gui.set_active(True)
self.assert_recording()
def test_shows_not_recording_after_toggle(self):
self.gui.set_active(True)
self.gui.set_active(False)
self.assert_not_recording()
def test_shows_not_recording_when_recording_finished(self):
self.gui.set_active(True)
self.message_broker.signal(MessageType.recording_finished)
self.assert_not_recording()
class TestStatusBar(ComponentBaseTest):
def setUp(self) -> None:
super(TestStatusBar, self).setUp()
self.gui = Gtk.Statusbar()
self.err_icon = Gtk.Image()
self.warn_icon = Gtk.Image()
self.statusbar = StatusBar(
self.message_broker,
self.controller_mock,
self.gui,
self.err_icon,
self.warn_icon,
)
self.message_broker.signal(MessageType.init)
def assert_empty(self):
self.assertFalse(self.err_icon.get_visible())
self.assertFalse(self.warn_icon.get_visible())
self.assertEqual(self.get_text(), "")
self.assertIsNone(self.get_tooltip())
def assert_error_status(self):
self.assertTrue(self.err_icon.get_visible())
self.assertFalse(self.warn_icon.get_visible())
def assert_warning_status(self):
self.assertFalse(self.err_icon.get_visible())
self.assertTrue(self.warn_icon.get_visible())
def get_text(self) -> str:
return self.gui.get_message_area().get_children()[0].get_text()
def get_tooltip(self) -> Optional[str]:
return self.gui.get_tooltip_text()
def test_starts_empty(self):
self.assert_empty()
def test_shows_error_status(self):
self.message_broker.send(StatusData(CTX_ERROR, "msg", "tooltip"))
self.assertEqual(self.get_text(), "msg")
self.assertEqual(self.get_tooltip(), "tooltip")
self.assert_error_status()
def test_shows_warning_status(self):
self.message_broker.send(StatusData(CTX_WARNING, "msg", "tooltip"))
self.assertEqual(self.get_text(), "msg")
self.assertEqual(self.get_tooltip(), "tooltip")
self.assert_warning_status()
def test_shows_newest_message(self):
self.message_broker.send(StatusData(CTX_ERROR, "msg", "tooltip"))
self.message_broker.send(StatusData(CTX_WARNING, "msg2", "tooltip2"))
self.assertEqual(self.get_text(), "msg2")
self.assertEqual(self.get_tooltip(), "tooltip2")
self.assert_warning_status()
def test_data_without_message_removes_messages(self):
self.message_broker.send(StatusData(CTX_WARNING, "msg", "tooltip"))
self.message_broker.send(StatusData(CTX_WARNING, "msg2", "tooltip2"))
self.message_broker.send(StatusData(CTX_WARNING))
self.assert_empty()
def test_restores_message_from_not_removed_ctx_id(self):
self.message_broker.send(StatusData(CTX_ERROR, "msg", "tooltip"))
self.message_broker.send(StatusData(CTX_WARNING, "msg2", "tooltip2"))
self.message_broker.send(StatusData(CTX_WARNING))
self.assertEqual(self.get_text(), "msg")
self.assert_error_status()
# works also the other way round
self.message_broker.send(StatusData(CTX_ERROR))
self.message_broker.send(StatusData(CTX_WARNING, "msg", "tooltip"))
self.message_broker.send(StatusData(CTX_ERROR, "msg2", "tooltip2"))
self.message_broker.send(StatusData(CTX_ERROR))
self.assertEqual(self.get_text(), "msg")
self.assert_warning_status()
def test_sets_msg_as_tooltip_if_tooltip_is_none(self):
self.message_broker.send(StatusData(CTX_ERROR, "msg"))
self.assertEqual(self.get_tooltip(), "msg")
class TestAutoloadSwitch(ComponentBaseTest):
def setUp(self) -> None:
super(TestAutoloadSwitch, self).setUp()
self.gui = Gtk.Switch()
self.switch = AutoloadSwitch(
self.message_broker, self.controller_mock, self.gui
)
def test_sets_autoload(self):
self.gui.set_active(True)
self.controller_mock.set_autoload.assert_called_once_with(True)
self.controller_mock.reset_mock()
self.gui.set_active(False)
self.controller_mock.set_autoload.assert_called_once_with(False)
def test_updates_state(self):
self.message_broker.send(PresetData(None, None, autoload=True))
self.assertTrue(self.gui.get_active())
self.message_broker.send(PresetData(None, None, autoload=False))
self.assertFalse(self.gui.get_active())
def test_avoids_infinite_recursion(self):
self.message_broker.send(PresetData(None, None, autoload=True))
self.message_broker.send(PresetData(None, None, autoload=False))
self.controller_mock.set_autoload.assert_not_called()
class TestReleaseCombinationSwitch(ComponentBaseTest):
def setUp(self) -> None:
super(TestReleaseCombinationSwitch, self).setUp()
self.gui = Gtk.Switch()
self.switch = ReleaseCombinationSwitch(
self.message_broker, self.controller_mock, self.gui
)
def test_updates_mapping(self):
self.gui.set_active(True)
self.controller_mock.update_mapping.assert_called_once_with(
release_combination_keys=True
)
self.controller_mock.reset_mock()
self.gui.set_active(False)
self.controller_mock.update_mapping.assert_called_once_with(
release_combination_keys=False
)
def test_updates_state(self):
self.message_broker.send(MappingData(release_combination_keys=True))
self.assertTrue(self.gui.get_active())
self.message_broker.send(MappingData(release_combination_keys=False))
self.assertFalse(self.gui.get_active())
def test_avoids_infinite_recursion(self):
self.message_broker.send(MappingData(release_combination_keys=True))
self.message_broker.send(MappingData(release_combination_keys=False))
self.controller_mock.update_mapping.assert_not_called()
class TestEventEntry(ComponentBaseTest):
def setUp(self) -> None:
super(TestEventEntry, self).setUp()
self.gui = EventEntry(InputEvent.from_string("3,0,1"), self.controller_mock)
def test_move_event(self):
self.gui._up_btn.clicked()
self.controller_mock.move_event_in_combination.assert_called_once_with(
InputEvent.from_string("3,0,1"), "up"
)
self.controller_mock.reset_mock()
self.gui._down_btn.clicked()
self.controller_mock.move_event_in_combination.assert_called_once_with(
InputEvent.from_string("3,0,1"), "down"
)
class TestCombinationListbox(ComponentBaseTest):
def setUp(self) -> None:
super(TestCombinationListbox, self).setUp()
self.gui = Gtk.ListBox()
self.listbox = CombinationListbox(
self.message_broker, self.controller_mock, self.gui
)
self.controller_mock.is_empty_mapping.return_value = False
self.message_broker.send(
MappingData(event_combination="1,1,1+3,0,1+1,2,1", target_uinput="keyboard")
)
def get_selected_row(self) -> EventEntry:
row = None
def find_row(r: EventEntry):
nonlocal row
if r.is_selected():
row = r
self.gui.foreach(find_row)
assert row is not None
return row
def select_row(self, event: InputEvent):
def select(row: EventEntry):
if row.input_event == event:
self.gui.select_row(row)
self.gui.foreach(select)
def test_loads_selected_row(self):
self.select_row(InputEvent.from_string("1,2,1"))
self.controller_mock.load_event.assert_called_once_with(
InputEvent.from_string("1,2,1")
)
def test_does_not_create_rows_when_mapping_is_empty(self):
self.controller_mock.is_empty_mapping.return_value = True
self.message_broker.send(MappingData(event_combination="1,1,1+3,0,1"))
self.assertEqual(len(self.gui.get_children()), 0)
def test_selects_row_when_selected_event_message_arrives(self):
self.message_broker.send(InputEvent.from_string("3,0,1"))
self.assertEqual(
self.get_selected_row().input_event, InputEvent.from_string("3,0,1")
)
def test_avoids_infinite_recursion(self):
self.message_broker.send(InputEvent.from_string("3,0,1"))
self.controller_mock.load_event.assert_not_called()
class TestAnalogInputSwitch(ComponentBaseTest):
def setUp(self) -> None:
super(TestAnalogInputSwitch, self).setUp()
self.gui = Gtk.Switch()
self.switch = AnalogInputSwitch(
self.message_broker, self.controller_mock, self.gui
)
def test_updates_event_as_analog(self):
self.gui.set_active(True)
self.controller_mock.set_event_as_analog.assert_called_once_with(True)
self.controller_mock.reset_mock()
self.gui.set_active(False)
self.controller_mock.set_event_as_analog.assert_called_once_with(False)
def test_updates_state(self):
self.message_broker.send(InputEvent.from_string("3,0,0"))
self.assertTrue(self.gui.get_active())
self.message_broker.send(InputEvent.from_string("3,0,10"))
self.assertFalse(self.gui.get_active())
def test_avoids_infinite_recursion(self):
self.message_broker.send(InputEvent.from_string("3,0,0"))
self.message_broker.send(InputEvent.from_string("3,0,-10"))
self.controller_mock.set_event_as_analog.assert_not_called()
def test_disables_switch_when_key_event(self):
self.message_broker.send(InputEvent.from_string("1,1,1"))
self.assertLess(self.gui.get_opacity(), 0.6)
self.assertFalse(self.gui.get_sensitive())
def test_enables_switch_when_axis_event(self):
self.message_broker.send(InputEvent.from_string("1,1,1"))
self.message_broker.send(InputEvent.from_string("3,0,10"))
self.assertEqual(self.gui.get_opacity(), 1)
self.assertTrue(self.gui.get_sensitive())
self.message_broker.send(InputEvent.from_string("1,1,1"))
self.message_broker.send(InputEvent.from_string("2,0,10"))
self.assertEqual(self.gui.get_opacity(), 1)
self.assertTrue(self.gui.get_sensitive())
class TestTriggerThresholdInput(ComponentBaseTest):
def setUp(self) -> None:
super(TestTriggerThresholdInput, self).setUp()
self.gui = Gtk.SpinButton()
self.input = TriggerThresholdInput(
self.message_broker, self.controller_mock, self.gui
)
self.message_broker.send(InputEvent.from_string("3,0,-10"))
def assert_abs_event_config(self):
self.assertEqual(self.gui.get_range(), (-99, 99))
self.assertTrue(self.gui.get_sensitive())
self.assertEqual(self.gui.get_opacity(), 1)
def assert_rel_event_config(self):
self.assertEqual(self.gui.get_range(), (-999, 999))
self.assertTrue(self.gui.get_sensitive())
self.assertEqual(self.gui.get_opacity(), 1)
def assert_key_event_config(self):
self.assertFalse(self.gui.get_sensitive())
self.assertLess(self.gui.get_opacity(), 0.6)
def test_updates_event(self):
self.gui.set_value(15)
self.controller_mock.update_event.assert_called_once_with(
InputEvent.from_string("3,0,15")
)
def test_sets_value_on_selected_event_message(self):
self.message_broker.send(InputEvent.from_string("3,0,10"))
self.assertEqual(self.gui.get_value(), 10)
def test_avoids_infinite_recursion(self):
self.message_broker.send(InputEvent.from_string("3,0,10"))
self.controller_mock.update_event.assert_not_called()
def test_updates_configuration_according_to_selected_event(self):
self.assert_abs_event_config()
self.message_broker.send(InputEvent.from_string("2,0,-10"))
self.assert_rel_event_config()
self.message_broker.send(InputEvent.from_string("1,1,1"))
self.assert_key_event_config()
class TestReleaseTimeoutInput(ComponentBaseTest):
def setUp(self) -> None:
super(TestReleaseTimeoutInput, self).setUp()
self.gui = Gtk.SpinButton()
self.input = ReleaseTimeoutInput(
self.message_broker, self.controller_mock, self.gui
)
self.message_broker.send(
MappingData(
event_combination=EventCombination("2,0,1"), target_uinput="keyboard"
)
)
def test_updates_timeout_on_mapping_message(self):
self.message_broker.send(
MappingData(event_combination=EventCombination("2,0,1"), release_timeout=1)
)
self.assertEqual(self.gui.get_value(), 1)
def test_updates_mapping(self):
self.gui.set_value(0.5)
self.controller_mock.update_mapping.assert_called_once_with(release_timeout=0.5)
def test_avoids_infinite_recursion(self):
self.message_broker.send(
MappingData(event_combination=EventCombination("2,0,1"), release_timeout=1)
)
self.controller_mock.update_mapping.assert_not_called()
def test_disables_input_based_on_input_combination(self):
self.message_broker.send(
MappingData(event_combination=EventCombination.from_string("2,0,1+1,1,1"))
)
self.assertTrue(self.gui.get_sensitive())
self.assertEqual(self.gui.get_opacity(), 1)
self.message_broker.send(
MappingData(event_combination=EventCombination.from_string("1,1,1+1,2,1"))
)
self.assertFalse(self.gui.get_sensitive())
self.assertLess(self.gui.get_opacity(), 0.6)
self.message_broker.send(
MappingData(event_combination=EventCombination.from_string("2,0,1+1,1,1"))
)
self.message_broker.send(
MappingData(event_combination=EventCombination.from_string("3,0,1+1,2,1"))
)
self.assertFalse(self.gui.get_sensitive())
self.assertLess(self.gui.get_opacity(), 0.6)
class TestOutputAxisSelector(ComponentBaseTest):
def setUp(self) -> None:
super(TestOutputAxisSelector, self).setUp()
self.gui = Gtk.ComboBox()
self.selection = OutputAxisSelector(
self.message_broker, self.controller_mock, self.gui
)
2022-07-25 08:55:26 +00:00
absinfo = evdev.AbsInfo(0, -10, 10, 0, 0, 0)
Frontend Refactor (#375) * 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>
2022-07-23 08:53:41 +00:00
self.message_broker.send(
UInputsData(
{
"mouse": {1: [1, 2, 3, 4], 2: [0, 1, 2, 3]},
"keyboard": {1: [1, 2, 3, 4]},
2022-07-25 08:55:26 +00:00
"gamepad": {
2: [0, 1, 2, 3],
3: [(0, absinfo), (1, absinfo), (2, absinfo), (3, absinfo)],
},
Frontend Refactor (#375) * 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>
2022-07-23 08:53:41 +00:00
}
)
)
self.message_broker.send(
MappingData(target_uinput="mouse", event_combination="1,1,1")
)
def set_active_selection(self, selection: Tuple):
self.gui.set_active_id(f"{selection[0]}, {selection[1]}")
def get_active_selection(self) -> Tuple[int, int]:
return tuple(int(i) for i in self.gui.get_active_id().split(",")) # type: ignore
def test_updates_mapping(self):
self.set_active_selection((2, 0))
self.controller_mock.update_mapping.assert_called_once_with(
output_type=2, output_code=0
)
def test_updates_mapping_with_none(self):
self.set_active_selection((2, 0))
self.controller_mock.reset_mock()
self.set_active_selection((None, None))
self.controller_mock.update_mapping.assert_called_once_with(
output_type=None, output_code=None
)
def test_selects_correct_entry(self):
self.assertEqual(self.gui.get_active_id(), "None, None")
self.message_broker.send(
MappingData(target_uinput="mouse", output_type=2, output_code=3)
)
self.assertEqual(self.get_active_selection(), (2, 3))
def test_avoids_infinite_recursion(self):
self.message_broker.send(
MappingData(target_uinput="mouse", output_type=2, output_code=3)
)
self.controller_mock.update_mapping.assert_not_called()
def test_updates_dropdown_model(self):
self.assertEqual(len(self.gui.get_model()), 5)
self.message_broker.send(MappingData(target_uinput="keyboard"))
self.assertEqual(len(self.gui.get_model()), 1)
self.message_broker.send(MappingData(target_uinput="gamepad"))
self.assertEqual(len(self.gui.get_model()), 9)
class TestKeyAxisStack(ComponentBaseTest):
def setUp(self) -> None:
super(TestKeyAxisStack, self).setUp()
self.gui = Gtk.Stack()
self.gui.add_named(Gtk.Box(), "Analog Axis")
self.gui.add_named(Gtk.Box(), "Key or Macro")
self.stack = KeyAxisStack(self.message_broker, self.controller_mock, self.gui)
self.gui.show_all()
self.gui.set_visible_child_name("Key or Macro")
def test_switches_to_axis_when_mapping_has_output_type_code_but_not_symbol(self):
self.message_broker.send(MappingData(output_type=2, output_code=0))
self.assertEqual(self.gui.get_visible_child_name(), "Analog Axis")
def test_switches_to_key_when_mapping_has_output_symbol_but_type_code(self):
self.gui.set_visible_child_name("Analog Axis")
self.message_broker.send(MappingData(output_symbol="a"))
self.assertEqual(self.gui.get_visible_child_name(), "Key or Macro")
def test_does_not_switch_when_mapping_is_ambiguous(self):
self.message_broker.send(
MappingData(output_type=2, output_code=0, output_symbol="a")
)
self.assertEqual(self.gui.get_visible_child_name(), "Key or Macro")
self.message_broker.send(MappingData(output_type=2, output_symbol="a"))
self.assertEqual(self.gui.get_visible_child_name(), "Key or Macro")
self.message_broker.send(MappingData(output_code=0, output_symbol="a"))
self.assertEqual(self.gui.get_visible_child_name(), "Key or Macro")
self.gui.set_visible_child_name("Analog Axis")
self.message_broker.send(
MappingData(output_type=2, output_code=0, output_symbol="a")
)
self.assertEqual(self.gui.get_visible_child_name(), "Analog Axis")
self.message_broker.send(MappingData(output_type=2, output_symbol="a"))
self.assertEqual(self.gui.get_visible_child_name(), "Analog Axis")
self.message_broker.send(MappingData(output_code=0, output_symbol="a"))
self.assertEqual(self.gui.get_visible_child_name(), "Analog Axis")
class TestTransformationDrawArea(ComponentBaseTest):
def setUp(self) -> None:
super(TestTransformationDrawArea, self).setUp()
self.gui = Gtk.Window()
self.draw_area = Gtk.DrawingArea()
self.gui.add(self.draw_area)
self.transform_draw_area = TransformationDrawArea(
self.message_broker, self.controller_mock, self.draw_area
)
def test_draws_transform(self):
with spy(self.transform_draw_area, "_transformation") as mock:
self.gui.show_all()
gtk_iteration()
mock.assert_called()
def test_updates_transform_when_mapping_updates(self):
old_tf = self.transform_draw_area._transformation
self.message_broker.send(MappingData(gain=2))
self.assertIsNot(old_tf, self.transform_draw_area._transformation)
def test_redraws_when_mapping_updates(self):
self.gui.show_all()
gtk_iteration(20)
mock = MagicMock()
self.draw_area.connect("draw", mock)
self.message_broker.send(MappingData(gain=2))
gtk_iteration(20)
mock.assert_called()
class TestSliders(ComponentBaseTest):
def setUp(self) -> None:
super(TestSliders, self).setUp()
self.gui = Gtk.Box()
self.gain = Gtk.Scale()
self.deadzone = Gtk.Scale()
self.expo = Gtk.Scale()
# add everything to a box: it will be cleand up properly
self.gui.add(self.gain)
self.gui.add(self.deadzone)
self.gui.add(self.expo)
self.sliders = Sliders(
self.message_broker,
self.controller_mock,
self.gain,
self.deadzone,
self.expo,
)
self.message_broker.send(
MappingData(event_combination="3,0,0", target_uinput="mouse")
)
@staticmethod
def get_range(range: Gtk.Range) -> Tuple[int, int]:
"""the Gtk.Range, has no get_range method. this is a workaround"""
v = range.get_value()
range.set_value(-(2**16))
min_ = range.get_value()
range.set_value(2**16)
max_ = range.get_value()
range.set_value(v)
return min_, max_
def test_slider_ranges(self):
self.assertEqual(self.get_range(self.gain), (-2, 2))
self.assertEqual(self.get_range(self.deadzone), (0, 0.9))
self.assertEqual(self.get_range(self.expo), (-1, 1))
def test_updates_value(self):
self.message_broker.send(
MappingData(
gain=0.5,
deadzone=0.6,
expo=0.3,
)
)
self.assertEqual(self.gain.get_value(), 0.5)
self.assertEqual(self.expo.get_value(), 0.3)
self.assertEqual(self.deadzone.get_value(), 0.6)
def test_gain_updates_mapping(self):
self.gain.set_value(0.5)
self.controller_mock.update_mapping.assert_called_once_with(gain=0.5)
def test_expo_updates_mapping(self):
self.expo.set_value(0.5)
self.controller_mock.update_mapping.assert_called_once_with(expo=0.5)
def test_deadzone_updates_mapping(self):
self.deadzone.set_value(0.5)
self.controller_mock.update_mapping.assert_called_once_with(deadzone=0.5)
def test_avoids_recursion(self):
self.message_broker.send(MappingData(gain=0.5))
self.controller_mock.update_mapping.assert_not_called()
self.message_broker.send(MappingData(expo=0.5))
self.controller_mock.update_mapping.assert_not_called()
self.message_broker.send(MappingData(deadzone=0.5))
self.controller_mock.update_mapping.assert_not_called()