diff --git a/qutebrowser/test/keyinput/test_basekeyparser.py b/qutebrowser/test/keyinput/test_basekeyparser.py new file mode 100644 index 000000000..3763cced3 --- /dev/null +++ b/qutebrowser/test/keyinput/test_basekeyparser.py @@ -0,0 +1,286 @@ +# Copyright 2014 Florian Bruhin (The Compiler) +# +# This file is part of qutebrowser. +# +# qutebrowser 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. +# +# qutebrowser 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 qutebrowser. If not, see . + +# pylint: disable=missing-docstring + +"""Tests for BaseKeyParser.""" + +import unittest +import logging +from unittest import TestCase +from unittest.mock import Mock + +import qutebrowser.keyinput._basekeyparser as basekeyparser + +from PyQt5.QtCore import Qt + + +def setUpModule(): + basekeyparser.QObject = Mock() + logging.disable(logging.ERROR) + + +class ConfigStub: + + """Stub for basekeyparser.config.""" + + DATA = {'test': {'': 'ctrla', + 'a': 'a', + 'ba': 'ba', + 'ax': 'ax', + 'ccc': 'ccc',}, + 'input': {'timeout': 100}, + 'test2': {'foo': 'bar', '': 'ctrlx'}} + + def section(self, name): + if name not in ['test', 'test2']: + raise ValueError("section called with section '{}'!".format(name)) + return self.DATA[name] + + def get(self, sect, opt): + return self.DATA[sect][opt] + + +class FakeKeyEvent: + + """Fake QKeyPressEvent stub.""" + + def __init__(self, key, modifiers=0, text=''): + self.key = Mock(return_value=key) + self.text = Mock(return_value=text) + self.modifiers = Mock(return_value=modifiers) + + +class NormalizeTests(TestCase): + + """Test _normalize_keystr method.""" + + def setUp(self): + self.kp = basekeyparser.BaseKeyParser() + + def test_normalize(self): + STRINGS = [ + ('Control+x', 'Ctrl+X'), + ('Windows+x', 'Meta+X'), + ('Mod1+x', 'Alt+X'), + ('Mod4+x', 'Meta+X'), + ('Control--', 'Ctrl+-'), + ('Windows++', 'Meta++'), + ] + for orig, repl in STRINGS: + self.assertEqual(self.kp._normalize_keystr(orig), repl, orig) + + +class SplitCountTests(TestCase): + + """Test the _split_count method.""" + + def setUp(self): + self.kp = basekeyparser.BaseKeyParser(supports_count=True) + + def test_onlycount(self): + self.kp._keystring = '10' + self.assertEqual(self.kp._split_count(), (10, '')) + + def test_normalcount(self): + self.kp._keystring = '10foo' + self.assertEqual(self.kp._split_count(), (10, 'foo')) + + def test_normalcount(self): + self.kp._keystring = '10foo' + self.assertEqual(self.kp._split_count(), (10, 'foo')) + + def test_minuscount(self): + self.kp._keystring = '-1foo' + self.assertEqual(self.kp._split_count(), (None, '-1foo')) + + def test_expcount(self): + self.kp._keystring = '10e4foo' + self.assertEqual(self.kp._split_count(), (10, 'e4foo')) + + def test_nocount(self): + self.kp._keystring = 'foo' + self.assertEqual(self.kp._split_count(), (None, 'foo')) + + def test_nosupport(self): + self.kp._supports_count = False + self.kp._keystring = '10foo' + self.assertEqual(self.kp._split_count(), (None, '10foo')) + + +class ReadConfigTests(TestCase): + + """Test reading the config.""" + + def setUp(self): + basekeyparser.config = ConfigStub() + basekeyparser.QTimer = Mock() + + def test_read_config_invalid(self): + kp = basekeyparser.BaseKeyParser() + with self.assertRaises(ValueError): + kp.read_config() + + def test_read_config_valid(self): + kp = basekeyparser.BaseKeyParser(supports_count=True, + supports_chains=True) + kp.read_config('test') + self.assertIn('ccc', kp.bindings) + self.assertIn('Ctrl+A', kp.special_bindings) + kp.read_config('test2') + self.assertNotIn('ccc', kp.bindings) + self.assertNotIn('Ctrl+A', kp.special_bindings) + self.assertIn('foo', kp.bindings) + self.assertIn('Ctrl+X', kp.special_bindings) + + +class SpecialKeysTests(TestCase): + + """Check execute() with special keys.""" + + def setUp(self): + basekeyparser.config = ConfigStub() + basekeyparser.QTimer = Mock() + self.kp = basekeyparser.BaseKeyParser() + self.kp.execute = Mock() + self.kp.read_config('test') + + def test_valid_key(self): + self.kp.handle(FakeKeyEvent(Qt.Key_A, Qt.ControlModifier)) + self.kp.handle(FakeKeyEvent(Qt.Key_X, Qt.ControlModifier)) + self.kp.execute.assert_called_once_with('ctrla', self.kp.Type.special) + + def test_invalid_key(self): + self.kp.handle(FakeKeyEvent(Qt.Key_A, Qt.ControlModifier | + Qt.AltModifier)) + self.assertFalse(self.kp.execute.called) + + def test_keychain(self): + self.kp.handle(FakeKeyEvent(Qt.Key_B)) + self.kp.handle(FakeKeyEvent(Qt.Key_A)) + self.assertFalse(self.kp.execute.called) + + +class KeyChainTests(TestCase): + + """Test execute() with keychain support.""" + + def setUp(self): + basekeyparser.config = ConfigStub() + self.timermock = Mock() + basekeyparser.QTimer = Mock(return_value=self.timermock) + self.kp = basekeyparser.BaseKeyParser(supports_chains=True, + supports_count=False) + self.kp.execute = Mock() + self.kp.read_config('test') + + def test_valid_special_key(self): + self.kp.handle(FakeKeyEvent(Qt.Key_A, Qt.ControlModifier)) + self.kp.handle(FakeKeyEvent(Qt.Key_X, Qt.ControlModifier)) + self.kp.execute.assert_called_once_with('ctrla', self.kp.Type.special) + self.assertEqual(self.kp._keystring, '') + + def test_invalid_special_key(self): + self.kp.handle(FakeKeyEvent(Qt.Key_A, Qt.ControlModifier | + Qt.AltModifier)) + self.assertFalse(self.kp.execute.called) + self.assertEqual(self.kp._keystring, '') + + def test_keychain(self): + self.kp.handle(FakeKeyEvent(Qt.Key_B, text='b')) + self.kp.handle(FakeKeyEvent(Qt.Key_A, text='a')) + self.kp.execute.assert_called_once_with('ba', self.kp.Type.chain, None) + self.assertEqual(self.kp._keystring, '') + + def test_ambigious_keychain(self): + # We start with 'a' where the keychain gives us an ambigious result. + # Then we check if the timer has been set up correctly + self.kp.handle(FakeKeyEvent(Qt.Key_A, text='a')) + self.assertFalse(self.kp.execute.called) + basekeyparser.QTimer.assert_called_once_with(self.kp) + self.timermock.setSingleShot.assert_called_once_with(True) + self.timermock.setInterval.assert_called_once_with(100) + self.assertTrue(self.timermock.timeout.connect.called) + self.assertFalse(self.timermock.stop.called) + self.timermock.start.assert_called_once_with() + # Now we type an 'x' and check 'ax' has been executed and the timer + # stopped. + self.kp.handle(FakeKeyEvent(Qt.Key_X, text='x')) + self.kp.execute.assert_called_once_with('ax', self.kp.Type.chain, None) + self.timermock.stop.assert_called_once_with() + self.assertEqual(self.kp._keystring, '') + + def test_invalid_keychain(self): + self.kp.handle(FakeKeyEvent(Qt.Key_B, text='b')) + self.kp.handle(FakeKeyEvent(Qt.Key_C, text='c')) + self.assertEqual(self.kp._keystring, '') + + +class CountTests(TestCase): + + """Test execute() with counts.""" + + def setUp(self): + basekeyparser.config = ConfigStub() + basekeyparser.QTimer = Mock() + self.kp = basekeyparser.BaseKeyParser(supports_chains=True, + supports_count=True) + self.kp.execute = Mock() + self.kp.read_config('test') + + def test_no_count(self): + self.kp.handle(FakeKeyEvent(Qt.Key_B, text='b')) + self.kp.handle(FakeKeyEvent(Qt.Key_A, text='a')) + self.kp.execute.assert_called_once_with('ba', self.kp.Type.chain, None) + self.assertEqual(self.kp._keystring, '') + + def test_count_0(self): + self.kp.handle(FakeKeyEvent(Qt.Key_0, text='0')) + self.kp.handle(FakeKeyEvent(Qt.Key_B, text='b')) + self.kp.handle(FakeKeyEvent(Qt.Key_A, text='a')) + self.kp.execute.assert_called_once_with('ba', self.kp.Type.chain, 0) + self.assertEqual(self.kp._keystring, '') + + def test_count_42(self): + self.kp.handle(FakeKeyEvent(Qt.Key_4, text='4')) + self.kp.handle(FakeKeyEvent(Qt.Key_2, text='2')) + self.kp.handle(FakeKeyEvent(Qt.Key_B, text='b')) + self.kp.handle(FakeKeyEvent(Qt.Key_A, text='a')) + self.kp.execute.assert_called_once_with('ba', self.kp.Type.chain, 42) + self.assertEqual(self.kp._keystring, '') + + def test_count_42_invalid(self): + # Invalid call with ccx gets ignored + self.kp.handle(FakeKeyEvent(Qt.Key_4, text='4')) + self.kp.handle(FakeKeyEvent(Qt.Key_2, text='2')) + self.kp.handle(FakeKeyEvent(Qt.Key_B, text='c')) + self.kp.handle(FakeKeyEvent(Qt.Key_A, text='c')) + self.kp.handle(FakeKeyEvent(Qt.Key_A, text='x')) + self.assertFalse(self.kp.execute.called) + self.assertEqual(self.kp._keystring, '') + # Valid call with ccc gets the correct count + self.kp.handle(FakeKeyEvent(Qt.Key_4, text='2')) + self.kp.handle(FakeKeyEvent(Qt.Key_2, text='3')) + self.kp.handle(FakeKeyEvent(Qt.Key_B, text='c')) + self.kp.handle(FakeKeyEvent(Qt.Key_A, text='c')) + self.kp.handle(FakeKeyEvent(Qt.Key_A, text='c')) + self.kp.execute.assert_called_once_with('ccc', self.kp.Type.chain, 23) + self.assertEqual(self.kp._keystring, '') + + +if __name__ == '__main__': + unittest.main()