Use signals and decorators for configs again.

This basically rolls back 64a119afb2 and
b7f2a6d143.

Fixes #188.
Breaks #156 again but the next commit will fix this.
This commit is contained in:
Florian Bruhin 2014-10-18 19:50:10 +02:00
parent bff0efb4a4
commit 76de3d0c51
15 changed files with 137 additions and 79 deletions

View File

@ -23,7 +23,7 @@ from PyQt5.QtNetwork import QNetworkCookie, QNetworkCookieJar
from PyQt5.QtCore import QStandardPaths, QDateTime
from qutebrowser.config import config, lineparser
from qutebrowser.utils import utils, standarddir
from qutebrowser.utils import utils, standarddir, objreg
class CookieJar(QNetworkCookieJar):
@ -39,8 +39,7 @@ class CookieJar(QNetworkCookieJar):
for line in self._linecp:
cookies += QNetworkCookie.parseCookies(line)
self.setAllCookies(cookies)
config.on_change(self.cookies_store_changed,
'permissions', 'cookies-store')
objreg.get('config').changed.connect(self.cookies_store_changed)
def __repr__(self):
return utils.get_repr(self, count=len(self.allCookies()))
@ -81,6 +80,7 @@ class CookieJar(QNetworkCookieJar):
self._linecp.data = lines
self._linecp.save()
@config.change_filter('permissions', 'cookies-store')
def cookies_store_changed(self):
"""Delete stored cookies if cookies-store changed."""
if not config.get('permissions', 'cookies-store'):

View File

@ -27,14 +27,12 @@ we borrow some methods and classes from there where it makes sense.
import os
import sys
import os.path
import inspect
import functools
import weakref
import configparser
import collections
import collections.abc
from PyQt5.QtCore import pyqtSignal, QObject, QStandardPaths
from PyQt5.QtCore import pyqtSignal, pyqtSlot, QObject, QStandardPaths
from PyQt5.QtWidgets import QMessageBox
from qutebrowser.config import (configdata, iniparsers, configtypes,
@ -44,31 +42,67 @@ from qutebrowser.utils import message, objreg, utils, standarddir, log
from qutebrowser.utils.usertypes import Completion
ChangeHandler = collections.namedtuple(
'ChangeHandler', ['func_ref', 'section', 'option'])
class change_filter: # pylint: disable=invalid-name
"""Decorator to register a new command handler.
change_handlers = []
This could also be a function, but as a class (with a "wrong" name) it's
much cleaner to implement.
Attributes:
_sectname: The section to be filtered.
_optname: The option to be filtered.
"""
def on_change(func, sectname=None, optname=None):
"""Register a new change handler.
def __init__(self, sectname, optname=None):
"""Save decorator arguments.
Gets called on parse-time with the decorator arguments.
Args:
func: The function to be called on change.
sectname: Filter for the config section.
If None, the handler gets called for all sections.
optname: Filter for the config option.
If None, the handler gets called for all options.
See class attributes.
"""
if optname is not None and sectname is None:
raise TypeError("option is {} but section is None!".format(optname))
if sectname is not None and sectname not in configdata.DATA:
raise NoSectionError("Section '{}' does not exist!".format(sectname))
if sectname not in configdata.DATA:
raise NoSectionError("Section '{}' does not exist!".format(
sectname))
if optname is not None and optname not in configdata.DATA[sectname]:
raise NoOptionError("Option '{}' does not exist in section "
"'{}'!".format(optname, sectname))
change_handlers.append(ChangeHandler(weakref.ref(func), sectname, optname))
self._sectname = sectname
self._optname = optname
def __call__(self, func):
"""Register the command before running the function.
Gets called when a function should be decorated.
Adds a filter which returns if we're not interested in the change-event
and calls the wrapped function if we are.
We assume the function passed doesn't take any parameters.
Args:
func: The function to be decorated.
Return:
The decorated function.
"""
@pyqtSlot(str, str)
@functools.wraps(func)
def wrapper(wrapper_self, sectname=None, optname=None):
# pylint: disable=missing-docstring
if sectname is None and optname is None:
# Called directly, not from a config change event.
return func(wrapper_self)
elif sectname != self._sectname:
return
elif self._optname is not None and optname != self._optname:
return
else:
return func(wrapper_self)
return wrapper
def get(*args, **kwargs):
@ -181,6 +215,7 @@ class ConfigManager(QObject):
_initialized: Whether the ConfigManager is fully initialized yet.
Signals:
changed: Emitted when a config option changed.
style_changed: When style caches need to be invalidated.
Args: the changed section and option.
"""
@ -188,6 +223,7 @@ class ConfigManager(QObject):
KEY_ESCAPE = r'\#['
ESCAPE_CHAR = '\\'
changed = pyqtSignal(str, str)
style_changed = pyqtSignal(str, str)
def __init__(self, configdir, fname, parent=None):
@ -316,28 +352,7 @@ class ConfigManager(QObject):
sectname, optname))
if sectname in ('colors', 'fonts'):
self.style_changed.emit(sectname, optname)
to_delete = []
for handler in change_handlers:
func = handler.func_ref()
if func is None:
to_delete.append(handler)
continue
elif handler.section is not None and handler.section != sectname:
continue
elif handler.option is not None and handler.option != optname:
continue
param_count = len(inspect.signature(func).parameters)
if param_count == 2:
func(sectname, optname)
elif param_count == 1:
func(sectname)
elif param_count == 0:
func()
else:
raise TypeError("Handler {} has invalid signature.".format(
utils.qualname(func)))
for handler in to_delete:
change_handlers.remove(handler)
self.changed.emit(sectname, optname)
def _after_set(self, changed_sect, changed_opt):
"""Clean up caches and emit signals after an option has been set."""

View File

@ -23,7 +23,9 @@ import os
import os.path
import collections
from qutebrowser.utils import log, utils
from PyQt5.QtCore import pyqtSlot
from qutebrowser.utils import log, utils, objreg
from qutebrowser.config import config
@ -37,6 +39,8 @@ class LineConfigParser(collections.UserList):
_configfile: The config file path.
_fname: Filename of the config.
_binary: Whether to open the file in binary mode.
_limit: The config section/option used to limit the maximum number of
lines.
"""
def __init__(self, configdir, fname, limit=None, binary=False):
@ -60,7 +64,7 @@ class LineConfigParser(collections.UserList):
log.init.debug("Reading config from {}".format(self._configfile))
self.read(self._configfile)
if limit is not None:
config.on_change(self.cleanup_file, *limit)
objreg.get('config').changed.connect(self.cleanup_file)
def __repr__(self):
return utils.get_repr(self, constructor=True,
@ -107,8 +111,11 @@ class LineConfigParser(collections.UserList):
with open(self._configfile, 'w', encoding='utf-8') as f:
self.write(f, limit)
@pyqtSlot(str, str)
def cleanup_file(self, section, option):
"""Delete the file if the limit was changed to 0."""
if (section, option) != self._limit:
return
value = config.get(section, option)
if value == 0:
if os.path.exists(self._configfile):

View File

@ -58,7 +58,8 @@ def set_register_stylesheet(obj):
log.style.vdebug("stylesheet for {}: {}".format(
obj.__class__.__name__, qss))
obj.setStyleSheet(qss)
config.on_change(functools.partial(update_stylesheet, obj))
objreg.get('config').changed.connect(
functools.partial(update_stylesheet, obj))
def update_stylesheet(obj):

View File

@ -32,7 +32,7 @@ from PyQt5.QtWebKit import QWebSettings
from PyQt5.QtCore import QStandardPaths
from qutebrowser.config import config
from qutebrowser.utils import usertypes, standarddir
from qutebrowser.utils import usertypes, standarddir, objreg
MapType = usertypes.enum('MapType', ['attribute', 'setter', 'static_setter'])
@ -192,7 +192,7 @@ def init():
for optname, (typ, arg) in section.items():
value = config.get(sectname, optname)
_set_setting(typ, arg, value)
config.on_change(update_settings)
objreg.get('config').changed.connect(update_settings)
def update_settings(section, option):

View File

@ -175,8 +175,7 @@ class ModeManager(QObject):
self._releaseevents_to_pass = []
self._forward_unbound_keys = config.get(
'input', 'forward-unbound-keys')
config.on_change(self.set_forward_unbound_keys, 'input',
'forward-unbound-keys')
objreg.get('config').changed.connect(self.set_forward_unbound_keys)
def __repr__(self):
return utils.get_repr(self, mode=self.mode(), locked=self.locked,
@ -331,6 +330,7 @@ class ModeManager(QObject):
raise ValueError("Can't leave normal mode!")
self.leave(self.mode(), 'leave current')
@config.change_filter('input', 'forward-unbound-keys')
def set_forward_unbound_keys(self):
"""Update local setting when config changed."""
self._forward_unbound_keys = config.get(

View File

@ -19,7 +19,7 @@
"""CompletionModels for different usages."""
from PyQt5.QtCore import Qt
from PyQt5.QtCore import pyqtSlot, Qt
from qutebrowser.config import config, configdata
from qutebrowser.models import basecompletion
@ -47,6 +47,7 @@ class SettingOptionCompletionModel(basecompletion.BaseCompletionModel):
Attributes:
_misc_items: A dict of the misc. column items which will be set later.
_section: The config section this model shows.
"""
# pylint: disable=abstract-method
@ -56,7 +57,8 @@ class SettingOptionCompletionModel(basecompletion.BaseCompletionModel):
cat = self.new_category(section)
sectdata = configdata.DATA[section]
self._misc_items = {}
config.on_change(self.update_misc_column, section)
self._section = section
objreg.get('config').changed.connect(self.update_misc_column)
for name in sectdata.keys():
try:
desc = sectdata.descriptions[name]
@ -71,8 +73,11 @@ class SettingOptionCompletionModel(basecompletion.BaseCompletionModel):
value)
self._misc_items[name] = miscitem
@pyqtSlot(str, str)
def update_misc_column(self, section, option):
"""Update misc column when config changed."""
if section != self._section:
return
try:
item = self._misc_items[option]
except KeyError:
@ -91,13 +96,20 @@ class SettingOptionCompletionModel(basecompletion.BaseCompletionModel):
class SettingValueCompletionModel(basecompletion.BaseCompletionModel):
"""A CompletionModel filled with setting values."""
"""A CompletionModel filled with setting values.
Attributes:
_section: The config section this model shows.
_option: The config option this model shows.
"""
# pylint: disable=abstract-method
def __init__(self, section, option=None, parent=None):
def __init__(self, section, option, parent=None):
super().__init__(parent)
config.on_change(self.update_current_value, section, option)
self._section = section
self._option = option
objreg.get('config').changed.connect(self.update_current_value)
cur_cat = self.new_category("Current", sort=0)
value = config.get(section, option, raw=True)
if not value:
@ -118,8 +130,11 @@ class SettingValueCompletionModel(basecompletion.BaseCompletionModel):
for (val, desc) in vals:
self.new_item(cat, val, desc)
@pyqtSlot(str, str)
def update_current_value(self, section, option):
"""Update current value when config changed."""
if (section, option) != (self._section, self._option):
return
value = config.get(section, option, raw=True)
if not value:
value = '""'

View File

@ -99,9 +99,9 @@ class CompletionView(QTreeView):
objreg.register('completer', completer_obj, scope='window',
window=win_id)
self.enabled = config.get('completion', 'show')
config.on_change(self.set_enabled, 'completion', 'show')
# FIXME
#config.on_change(self.init_command_completion, 'aliases')
objreg.get('config').changed.connect(self.set_enabled)
# FIXME handle new aliases.
#objreg.get('config').changed.connect(self.init_command_completion)
self._delegate = completiondelegate.CompletionItemDelegate(self)
self.setItemDelegate(self._delegate)
@ -206,6 +206,7 @@ class CompletionView(QTreeView):
if config.get('completion', 'shrink'):
self.resize_completion.emit()
@config.change_filter('completion', 'show')
def set_enabled(self):
"""Update self.enabled when the config changed."""
self.enabled = config.get('completion', 'show')

View File

@ -57,7 +57,7 @@ class ConsoleLineEdit(misc.CommandLineEdit):
"""
super().__init__(parent)
self.update_font()
config.on_change(self.update_font, 'fonts', 'debug-console')
objreg.get('config').changed.connect(self.update_font)
self.textChanged.connect(self.on_text_changed)
self._rlcompleter = rlcompleter.Completer(namespace)
@ -135,6 +135,7 @@ class ConsoleLineEdit(misc.CommandLineEdit):
else:
super().keyPressEvent(e)
@config.change_filter('fonts', 'debug-console')
def update_font(self):
"""Set the correct font."""
self.setFont(config.get('fonts', 'debug-console'))
@ -148,13 +149,14 @@ class ConsoleTextEdit(QTextEdit):
super().__init__(parent)
self.setAcceptRichText(False)
self.setReadOnly(True)
config.on_change(self.update_font, 'fonts', 'debug-console')
objreg.get('config').changed.connect(self.update_font)
self.update_font()
self.setFocusPolicy(Qt.ClickFocus)
def __repr__(self):
return utils.get_repr(self)
@config.change_filter('fonts', 'debug-console')
def update_font(self):
"""Update font when config changed."""
self.setFont(config.get('fonts', 'debug-console'))

View File

@ -116,9 +116,7 @@ class MainWindow(QWidget):
# resizing will fail. Therefore, we use singleShot QTimers to make sure
# we defer this until everything else is initialized.
QTimer.singleShot(0, self._connect_resize_completion)
config.on_change(self.resize_completion, 'completion', 'height')
config.on_change(self.resize_completion, 'completion', 'shrink')
objreg.get('config').changed.connect(self.on_config_changed)
#self.retranslateUi(MainWindow)
#self.tabWidget.setCurrentIndex(0)
#QtCore.QMetaObject.connectSlotsByName(MainWindow)
@ -126,6 +124,12 @@ class MainWindow(QWidget):
def __repr__(self):
return utils.get_repr(self)
@pyqtSlot(str, str)
def on_config_changed(self, section, option):
"""Resize the completion if related config options changed."""
if section == 'completion' and option in ('height', 'shrink'):
self.resize_completion()
@classmethod
def spawn(cls, show=True):
"""Create a new main window.

View File

@ -145,7 +145,7 @@ class StatusBar(QWidget):
self._text_pop_timer = usertypes.Timer(self, 'statusbar_text_pop')
self._text_pop_timer.timeout.connect(self._pop_text)
self.set_pop_timer_interval()
config.on_change(self.set_pop_timer_interval, 'ui', 'message-timeout')
objreg.get('config').changed.connect(self.set_pop_timer_interval)
self.prompt = prompt.Prompt(win_id)
self._stack.addWidget(self.prompt)
@ -402,6 +402,7 @@ class StatusBar(QWidget):
if mode == usertypes.KeyMode.insert:
self._set_insert_active(False)
@config.change_filter('ui', 'message-timeout')
def set_pop_timer_interval(self):
"""Update message timeout when config changed."""
self._text_pop_timer.setInterval(config.get('ui', 'message-timeout'))

View File

@ -23,7 +23,7 @@ from PyQt5.QtCore import pyqtSlot
from qutebrowser.config import config
from qutebrowser.widgets.statusbar import textbase
from qutebrowser.utils import usertypes, log
from qutebrowser.utils import usertypes, log, objreg
class Text(textbase.TextBase):
@ -46,7 +46,7 @@ class Text(textbase.TextBase):
self._normaltext = ''
self._temptext = ''
self._jstext = ''
config.on_change(self.update_text, 'ui', 'display-statusbar-messages')
objreg.get('config').changed.connect(self.update_text)
def set_text(self, which, text):
"""Set a text.
@ -76,6 +76,7 @@ class Text(textbase.TextBase):
else:
log.misc.debug("Ignoring reset: '{}'".format(text))
@config.change_filter('ui', 'display-statusbar-messages')
def update_text(self):
"""Update QLabel text when needed."""
if self._temptext:

View File

@ -116,7 +116,7 @@ class TabbedBrowser(tabwidget.TabWidget):
# FIXME adjust this to font size
# https://github.com/The-Compiler/qutebrowser/issues/119
self.setIconSize(QSize(12, 12))
config.on_change(self.update_favicons, 'tabs', 'show-favicons')
objreg.get('config').changed.connect(self.update_favicons)
def __repr__(self):
return utils.get_repr(self, count=self.count())
@ -386,6 +386,7 @@ class TabbedBrowser(tabwidget.TabWidget):
# We first want QWebPage to refresh.
QTimer.singleShot(0, check_scroll_pos)
@config.change_filter('tabs', 'show-favicons')
def update_favicons(self):
"""Update favicons when config was changed."""
show = config.get('tabs', 'show-favicons')

View File

@ -53,8 +53,9 @@ class TabWidget(QTabWidget):
self.setUsesScrollButtons(True)
bar.setDrawBase(False)
self.init_config()
config.on_change(self.init_config, 'tabs')
objreg.get('config').changed.connect(self.init_config)
@config.change_filter('tabs')
def init_config(self):
"""Initialize attributes based on the config."""
tabbar = self.tabBar()
@ -88,17 +89,19 @@ class TabBar(QTabBar):
self._win_id = win_id
self.setStyle(TabBarStyle(self.style()))
self.set_font()
config.on_change(self.set_font, 'fonts', 'tabbar')
config_obj = objreg.get('config')
config_obj.changed.connect(self.set_font)
self.vertical = False
self.setAutoFillBackground(True)
self.set_colors()
config.on_change(self.set_colors, 'colors', 'tab.bg.bar')
config_obj.changed.connect(self.set_colors)
QTimer.singleShot(0, self.autohide)
config.on_change(self.autohide, 'tabs', 'auto-hide')
config_obj.changed.connect(self.autohide)
def __repr__(self):
return utils.get_repr(self, count=self.count())
@config.change_filter('tabs', 'auto-hide')
def autohide(self):
"""Auto-hide the tabbar if needed."""
auto_hide = config.get('tabs', 'auto-hide')
@ -123,10 +126,12 @@ class TabBar(QTabBar):
self.setTabData(idx, color)
self.update(self.tabRect(idx))
@config.change_filter('fonts', 'tabbar')
def set_font(self):
"""Set the tabbar font."""
self.setFont(config.get('fonts', 'tabbar'))
@config.change_filter('colors', 'tab.bg.bar')
def set_colors(self):
"""Set the tabbar colors."""
p = self.palette()

View File

@ -95,8 +95,7 @@ class WebView(QWebView):
self._zoom = None
self._has_ssl_errors = False
self.init_neighborlist()
config.on_change(self.init_neighborlist, 'ui', 'zoom-levels')
config.on_change(self.init_neighborlist, 'ui', 'default-zoom')
objreg.get('config').changed.connect(self.init_neighborlist)
self._cur_url = None
self.cur_url = QUrl()
self.progress = 0
@ -142,6 +141,12 @@ class WebView(QWebView):
self.load_status = val
self.load_status_changed.emit(val.name)
@pyqtSlot(str, str)
def on_config_changed(self, section, option):
"""Reinitialize the zoom neighborlist if related config changed."""
if section == 'ui' and option in ('zoom-levels', 'default-zoom'):
self.init_neighborlist()
def init_neighborlist(self):
"""Initialize the _zoom neighborlist."""
levels = config.get('ui', 'zoom-levels')