# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et: # Copyright 2016 Ryan Roden-Corrent (rcorre) # # 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 . """Test the keyhint widget.""" from collections import OrderedDict import pytest from qutebrowser.misc.keyhintwidget import KeyHintView def expected_text(*args): """Helper to format text we expect the KeyHintView to generate. Args: args: One tuple for each row in the expected output. Tuples are of the form: (prefix, color, suffix, command). """ text = '' for group in args: text += ("" "" "" "" "").format(*group) return text + '
{}{}{}
' @pytest.fixture def keyhint_config(config_stub): """Fixture providing the necessary config settings for the KeyHintView.""" config_stub.data = { 'colors': { 'keyhint.fg': 'white', 'keyhint.fg.suffix': 'yellow', 'keyhint.bg': 'black', }, 'fonts': {'keyhint': 'Comic Sans'}, 'ui': { 'keyhint-blacklist': '', 'keyhint-delay': 500, 'status-position': 'bottom', }, } return config_stub @pytest.fixture def keyhint(qtbot, keyhint_config, key_config_stub): """Fixture to initialize a KeyHintView.""" keyhint = KeyHintView(0, None) qtbot.add_widget(keyhint) assert keyhint.text() == '' return keyhint def test_show_and_hide(qtbot, keyhint): with qtbot.waitSignal(keyhint.update_geometry): with qtbot.waitExposed(keyhint): keyhint.show() keyhint.update_keyhint('normal', '') assert not keyhint.isVisible() def test_position_change(keyhint, config_stub): config_stub.set('ui', 'status-position', 'top') stylesheet = keyhint.styleSheet() assert 'border-bottom-right-radius' in stylesheet assert 'border-top-right-radius' not in stylesheet def test_suggestions(keyhint, key_config_stub): """Test that keyhints are shown based on a prefix.""" # we want the dict to return sorted items() for reliable testing key_config_stub.set_bindings_for('normal', OrderedDict([ ('aa', 'cmd-aa'), ('ab', 'cmd-ab'), ('aba', 'cmd-aba'), ('abb', 'cmd-abb'), ('xd', 'cmd-xd'), ('xe', 'cmd-xe')])) keyhint.update_keyhint('normal', 'a') assert keyhint.text() == expected_text( ('a', 'yellow', 'a', 'cmd-aa'), ('a', 'yellow', 'b', 'cmd-ab'), ('a', 'yellow', 'ba', 'cmd-aba'), ('a', 'yellow', 'bb', 'cmd-abb')) def test_special_bindings(keyhint, key_config_stub): """Ensure a prefix of '<' doesn't suggest special keys.""" # we want the dict to return sorted items() for reliable testing key_config_stub.set_bindings_for('normal', OrderedDict([ ('', 'cmd-ctrla')])) keyhint.update_keyhint('normal', '<') assert keyhint.text() == expected_text( ('<', 'yellow', 'a', 'cmd-<a'), ('<', 'yellow', 'b', 'cmd-<b')) def test_color_switch(keyhint, config_stub, key_config_stub): """Ensure the keyhint suffix color can be updated at runtime.""" config_stub.set('colors', 'keyhint.fg.suffix', '#ABCDEF') key_config_stub.set_bindings_for('normal', OrderedDict([ ('aa', 'cmd-aa')])) keyhint.update_keyhint('normal', 'a') assert keyhint.text() == expected_text(('a', '#ABCDEF', 'a', 'cmd-aa')) def test_no_matches(keyhint, key_config_stub): """Ensure the widget isn't visible if there are no keystrings to show.""" key_config_stub.set_bindings_for('normal', OrderedDict([ ('aa', 'cmd-aa'), ('ab', 'cmd-ab')])) keyhint.update_keyhint('normal', 'z') assert not keyhint.text() assert not keyhint.isVisible() def test_blacklist(keyhint, config_stub, key_config_stub): """Test that blacklisted keychains aren't hinted.""" config_stub.set('ui', 'keyhint-blacklist', ['ab*']) # we want the dict to return sorted items() for reliable testing key_config_stub.set_bindings_for('normal', OrderedDict([ ('aa', 'cmd-aa'), ('ab', 'cmd-ab'), ('aba', 'cmd-aba'), ('abb', 'cmd-abb'), ('xd', 'cmd-xd'), ('xe', 'cmd-xe')])) keyhint.update_keyhint('normal', 'a') assert keyhint.text() == expected_text(('a', 'yellow', 'a', 'cmd-aa')) def test_blacklist_all(keyhint, config_stub, key_config_stub): """Test that setting the blacklist to * disables keyhints.""" config_stub.set('ui', 'keyhint-blacklist', ['*']) # we want the dict to return sorted items() for reliable testing key_config_stub.set_bindings_for('normal', OrderedDict([ ('aa', 'cmd-aa'), ('ab', 'cmd-ab'), ('aba', 'cmd-aba'), ('abb', 'cmd-abb'), ('xd', 'cmd-xd'), ('xe', 'cmd-xe')])) keyhint.update_keyhint('normal', 'a') assert not keyhint.text() def test_delay(qtbot, stubs, monkeypatch, keyhint_config, key_config_stub): timer = stubs.FakeTimer() monkeypatch.setattr( 'qutebrowser.misc.keyhintwidget.usertypes.Timer', lambda *_: timer) interval = 200 keyhint_config.set('ui', 'keyhint-delay', interval) key_config_stub.set_bindings_for('normal', OrderedDict([('aa', 'cmd-aa')])) keyhint = KeyHintView(0, None) keyhint.update_keyhint('normal', 'a') assert timer.interval() == interval