Implement a hybrid list/sql completion model.

Now all completion models are of a single type called CompletionModel.
This model combines one or more categories. A category can either be a
ListCategory or a SqlCategory.

This simplifies the API, and will allow the use of models that combine simple
list-based and sql sources. This is important for two reasons:

- Adding searchengines to url completion
- Using an on-disk sqlite database for history, while keeping bookmarks and
  quickmars as text files.
This commit is contained in:
Ryan Roden-Corrent 2017-02-22 22:25:11 -05:00
parent 921211bbaa
commit e3a33ca427
15 changed files with 426 additions and 407 deletions

View File

@ -24,7 +24,7 @@ from PyQt5.QtCore import pyqtSlot, QObject, QTimer
from qutebrowser.config import config from qutebrowser.config import config
from qutebrowser.commands import cmdutils, runners from qutebrowser.commands import cmdutils, runners
from qutebrowser.utils import log, utils from qutebrowser.utils import log, utils
from qutebrowser.completion.models import sortfilter, miscmodels from qutebrowser.completion.models import miscmodels
class Completer(QObject): class Completer(QObject):
@ -62,22 +62,6 @@ class Completer(QObject):
completion = self.parent() completion = self.parent()
return completion.model() return completion.model()
def _get_completion_model(self, completion, pos_args):
"""Get a completion model based on an enum member.
Args:
completion: A usertypes.Completion member.
pos_args: The positional args entered before the cursor.
Return:
A completion model or None.
"""
model = completion(*pos_args)
if model is None or hasattr(model, 'set_pattern'):
return model
else:
return sortfilter.CompletionFilterModel(source=model, parent=self)
def _get_new_completion(self, before_cursor, under_cursor): def _get_new_completion(self, before_cursor, under_cursor):
"""Get a new completion. """Get a new completion.
@ -96,9 +80,8 @@ class Completer(QObject):
log.completion.debug("After removing flags: {}".format(before_cursor)) log.completion.debug("After removing flags: {}".format(before_cursor))
if not before_cursor: if not before_cursor:
# '|' or 'set|' # '|' or 'set|'
model = miscmodels.command()
log.completion.debug('Starting command completion') log.completion.debug('Starting command completion')
return sortfilter.CompletionFilterModel(source=model, parent=self) return miscmodels.command()
try: try:
cmd = cmdutils.cmd_dict[before_cursor[0]] cmd = cmdutils.cmd_dict[before_cursor[0]]
except KeyError: except KeyError:
@ -113,7 +96,8 @@ class Completer(QObject):
return None return None
if completion is None: if completion is None:
return None return None
model = self._get_completion_model(completion, before_cursor[1:])
model = completion(*before_cursor[1:])
log.completion.debug('Starting {} completion' log.completion.debug('Starting {} completion'
.format(completion.__name__)) .format(completion.__name__))
return model return model

View File

@ -197,7 +197,7 @@ class CompletionItemDelegate(QStyledItemDelegate):
if index.parent().isValid(): if index.parent().isValid():
pattern = index.model().pattern pattern = index.model().pattern
columns_to_filter = index.model().srcmodel.columns_to_filter columns_to_filter = index.model().columns_to_filter
if index.column() in columns_to_filter and pattern: if index.column() in columns_to_filter and pattern:
repl = r'<span class="highlight">\g<0></span>' repl = r'<span class="highlight">\g<0></span>'
text = re.sub(re.escape(pattern).replace(r'\ ', r'|'), text = re.sub(re.escape(pattern).replace(r'\ ', r'|'),

View File

@ -299,7 +299,7 @@ class CompletionView(QTreeView):
if pattern is not None: if pattern is not None:
model.set_pattern(pattern) model.set_pattern(pattern)
self._column_widths = model.srcmodel.column_widths self._column_widths = model.column_widths
self._resize_columns() self._resize_columns()
self._maybe_update_geometry() self._maybe_update_geometry()
@ -368,7 +368,7 @@ class CompletionView(QTreeView):
"""Delete the current completion item.""" """Delete the current completion item."""
if not self.currentIndex().isValid(): if not self.currentIndex().isValid():
raise cmdexc.CommandError("No item selected!") raise cmdexc.CommandError("No item selected!")
if self.model().srcmodel.delete_cur_item is None: if self.model().delete_cur_item is None:
raise cmdexc.CommandError("Cannot delete this item.") raise cmdexc.CommandError("Cannot delete this item.")
else: else:
self.model().srcmodel.delete_cur_item(self) self.model().delete_cur_item(self)

View File

@ -1,105 +0,0 @@
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
# Copyright 2014-2017 Florian Bruhin (The Compiler) <mail@qutebrowser.org>
#
# 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 <http://www.gnu.org/licenses/>.
"""The base completion model for completion in the command line.
Module attributes:
Role: An enum of user defined model roles.
"""
from PyQt5.QtCore import Qt
from PyQt5.QtGui import QStandardItemModel, QStandardItem
class CompletionModel(QStandardItemModel):
"""A simple QStandardItemModel adopted for completions.
Used for showing completions later in the CompletionView. Supports setting
marks and adding new categories/items easily.
Attributes:
column_widths: The width percentages of the columns used in the
completion view.
"""
def __init__(self, *, column_widths=(30, 70, 0), columns_to_filter=None,
delete_cur_item=None, parent=None):
super().__init__(parent)
self.setColumnCount(3)
self.columns_to_filter = columns_to_filter or [0]
self.column_widths = column_widths
self.delete_cur_item = delete_cur_item
def new_category(self, name):
"""Add a new category to the model.
Args:
name: The name of the category to add.
Return:
The created QStandardItem.
"""
cat = QStandardItem(name)
self.appendRow(cat)
return cat
def new_item(self, cat, name, desc='', misc=None):
"""Add a new item to a category.
Args:
cat: The parent category.
name: The name of the item.
desc: The description of the item.
misc: Misc text to display.
Return:
A (nameitem, descitem, miscitem) tuple.
"""
assert not isinstance(name, int)
assert not isinstance(desc, int)
assert not isinstance(misc, int)
nameitem = QStandardItem(name)
descitem = QStandardItem(desc)
miscitem = QStandardItem(misc)
cat.appendRow([nameitem, descitem, miscitem])
def flags(self, index):
"""Return the item flags for index.
Override QAbstractItemModel::flags.
Args:
index: The QModelIndex to get item flags for.
Return:
The item flags, or Qt.NoItemFlags on error.
"""
if not index.isValid():
return
if index.parent().isValid():
# item
return (Qt.ItemIsEnabled | Qt.ItemIsSelectable |
Qt.ItemNeverHasChildren)
else:
# category
return Qt.NoItemFlags

View File

@ -1,6 +1,6 @@
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et: # vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
# Copyright 2016-2017 Ryan Roden-Corrent (rcorre) <ryan@rcorre.net> # Copyright 2017 Ryan Roden-Corrent (rcorre) <ryan@rcorre.net>
# #
# This file is part of qutebrowser. # This file is part of qutebrowser.
# #
@ -17,65 +17,29 @@
# You should have received a copy of the GNU General Public License # You should have received a copy of the GNU General Public License
# along with qutebrowser. If not, see <http://www.gnu.org/licenses/>. # along with qutebrowser. If not, see <http://www.gnu.org/licenses/>.
"""A completion model backed by SQL tables.""" """A model that proxies access to one or more completion categories."""
import re import re
from PyQt5.QtCore import Qt, QModelIndex, QAbstractItemModel from PyQt5.QtCore import Qt, QModelIndex, QAbstractItemModel
from PyQt5.QtSql import QSqlQueryModel
from qutebrowser.utils import log, qtutils from qutebrowser.utils import log, qtutils
from qutebrowser.misc import sql from qutebrowser.completion.models import sortfilter, listcategory, sqlcategory
class _SqlCompletionCategory(QSqlQueryModel): class CompletionModel(QAbstractItemModel):
"""Wraps a SqlQuery for use as a completion category."""
def __init__(self, name, *, sort_by, sort_order, select, where, """A model that proxies access to one or more completion categories.
columns_to_filter, parent=None):
super().__init__(parent=parent)
self.tablename = name
query = sql.run_query('select * from {} limit 1'.format(name)) Top level indices represent categories.
self._fields = [query.record().fieldName(i) for i in columns_to_filter] Child indices represent rows of those tables.
querystr = 'select {} from {} where ('.format(select, name)
# the incoming pattern will have literal % and _ escaped with '\'
# we need to tell sql to treat '\' as an escape character
querystr += ' or '.join("{} like ? escape '\\'".format(f)
for f in self._fields)
querystr += ')'
if where:
querystr += ' and ' + where
if sort_by:
assert sort_order == 'asc' or sort_order == 'desc'
querystr += ' order by {} {}'.format(sort_by, sort_order)
self._querystr = querystr
self.set_pattern('%')
def set_pattern(self, pattern):
query = sql.run_query(self._querystr, [pattern for _ in self._fields])
self.setQuery(query)
class SqlCompletionModel(QAbstractItemModel):
"""A sqlite-based model that provides data for the CompletionView.
This model is a wrapper around one or more sql tables. The tables are all
stored in a single database in qutebrowser's cache directory.
Top level indices represent categories, each of which is backed by a single
table. Child indices represent rows of those tables.
Attributes: Attributes:
column_widths: The width percentages of the columns used in the column_widths: The width percentages of the columns used in the
completion view. completion view.
columns_to_filter: A list of indices of columns to apply the filter to. columns_to_filter: A list of indices of columns to apply the filter to.
pattern: Current filter pattern, used for highlighting. pattern: Current filter pattern, used for highlighting.
_categories: The category tables. _categories: The sub-categories.
""" """
def __init__(self, *, column_widths=(30, 70, 0), columns_to_filter=None, def __init__(self, *, column_widths=(30, 70, 0), columns_to_filter=None,
@ -84,7 +48,6 @@ class SqlCompletionModel(QAbstractItemModel):
self.columns_to_filter = columns_to_filter or [0] self.columns_to_filter = columns_to_filter or [0]
self.column_widths = column_widths self.column_widths = column_widths
self._categories = [] self._categories = []
self.srcmodel = self # TODO: dummy for compat with old API
self.pattern = '' self.pattern = ''
self.delete_cur_item = delete_cur_item self.delete_cur_item = delete_cur_item
@ -94,7 +57,7 @@ class SqlCompletionModel(QAbstractItemModel):
Args: Args:
idx: A QModelIndex idx: A QModelIndex
Returns: Returns:
A _SqlCompletionCategory if the index points at one, else None A category if the index points at one, else None
""" """
# items hold an index to the parent category in their internalPointer # items hold an index to the parent category in their internalPointer
# categories have an empty internalPointer # categories have an empty internalPointer
@ -102,7 +65,19 @@ class SqlCompletionModel(QAbstractItemModel):
return self._categories[index.row()] return self._categories[index.row()]
return None return None
def new_category(self, name, *, select='*', where=None, sort_by=None, def add_list(self, name, items):
"""Add a list of items as a completion category.
Args:
name: Title of the category.
items: List of tuples.
"""
cat = listcategory.ListCategory(name, items, parent=self)
filtermodel = sortfilter.CompletionFilterModel(cat,
self.columns_to_filter)
self._categories.append(filtermodel)
def add_sqltable(self, name, *, select='*', where=None, sort_by=None,
sort_order=None): sort_order=None):
"""Create a new completion category and add it to this model. """Create a new completion category and add it to this model.
@ -112,10 +87,8 @@ class SqlCompletionModel(QAbstractItemModel):
where: An optional clause to filter out some rows. where: An optional clause to filter out some rows.
sort_by: The name of the field to sort by, or None for no sorting. sort_by: The name of the field to sort by, or None for no sorting.
sort_order: Either 'asc' or 'desc', if sort_by is non-None sort_order: Either 'asc' or 'desc', if sort_by is non-None
Return: A new CompletionCategory.
""" """
cat = _SqlCompletionCategory(name, parent=self, sort_by=sort_by, cat = sqlcategory.SqlCategory(name, parent=self, sort_by=sort_by,
sort_order=sort_order, sort_order=sort_order,
select=select, where=where, select=select, where=where,
columns_to_filter=self.columns_to_filter) columns_to_filter=self.columns_to_filter)
@ -135,11 +108,11 @@ class SqlCompletionModel(QAbstractItemModel):
return None return None
if not index.parent().isValid(): if not index.parent().isValid():
if index.column() == 0: if index.column() == 0:
return self._categories[index.row()].tablename return self._categories[index.row()].name
else: else:
table = self._categories[index.parent().row()] cat = self._categories[index.parent().row()]
idx = table.index(index.row(), index.column()) idx = cat.index(index.row(), index.column())
return table.data(idx) return cat.data(idx)
def flags(self, index): def flags(self, index):
"""Return the item flags for index. """Return the item flags for index.
@ -171,7 +144,7 @@ class SqlCompletionModel(QAbstractItemModel):
if parent.isValid(): if parent.isValid():
if parent.column() != 0: if parent.column() != 0:
return QModelIndex() return QModelIndex()
# store a pointer to the parent table in internalPointer # store a pointer to the parent category in internalPointer
return self.createIndex(row, col, self._categories[parent.row()]) return self.createIndex(row, col, self._categories[parent.row()])
return self.createIndex(row, col, None) return self.createIndex(row, col, None)
@ -183,11 +156,11 @@ class SqlCompletionModel(QAbstractItemModel):
Args: Args:
index: The QModelIndex to get the parent index for. index: The QModelIndex to get the parent index for.
""" """
parent_table = index.internalPointer() parent_cat = index.internalPointer()
if not parent_table: if not parent_cat:
# categories have no parent # categories have no parent
return QModelIndex() return QModelIndex()
row = self._categories.index(parent_table) row = self._categories.index(parent_cat)
return self.createIndex(row, 0, None) return self.createIndex(row, 0, None)
def rowCount(self, parent=QModelIndex()): def rowCount(self, parent=QModelIndex()):
@ -209,24 +182,24 @@ class SqlCompletionModel(QAbstractItemModel):
return 3 return 3
def canFetchMore(self, parent): def canFetchMore(self, parent):
"""Override to forward the call to the tables.""" """Override to forward the call to the categories."""
cat = self._cat_from_idx(parent) cat = self._cat_from_idx(parent)
if cat: if cat:
return cat.canFetchMore() return cat.canFetchMore(parent)
return False return False
def fetchMore(self, parent): def fetchMore(self, parent):
"""Override to forward the call to the tables.""" """Override to forward the call to the categories."""
cat = self._cat_from_idx(parent) cat = self._cat_from_idx(parent)
if cat: if cat:
cat.fetchMore() cat.fetchMore(parent)
def count(self): def count(self):
"""Return the count of non-category items.""" """Return the count of non-category items."""
return sum(t.rowCount() for t in self._categories) return sum(t.rowCount() for t in self._categories)
def set_pattern(self, pattern): def set_pattern(self, pattern):
"""Set the filter pattern for all category tables. """Set the filter pattern for all categories.
This will apply to the fields indicated in columns_to_filter. This will apply to the fields indicated in columns_to_filter.
@ -236,19 +209,13 @@ class SqlCompletionModel(QAbstractItemModel):
log.completion.debug("Setting completion pattern '{}'".format(pattern)) log.completion.debug("Setting completion pattern '{}'".format(pattern))
# TODO: should pattern be saved in the view layer instead? # TODO: should pattern be saved in the view layer instead?
self.pattern = pattern self.pattern = pattern
# escape to treat a user input % or _ as a literal, not a wildcard
pattern = pattern.replace('%', '\\%')
pattern = pattern.replace('_', '\\_')
# treat spaces as wildcards to match any of the typed words
pattern = re.sub(r' +', '%', pattern)
pattern = '%{}%'.format(pattern)
for cat in self._categories: for cat in self._categories:
cat.set_pattern(pattern) cat.set_pattern(pattern)
def first_item(self): def first_item(self):
"""Return the index of the first child (non-category) in the model.""" """Return the index of the first child (non-category) in the model."""
for row, table in enumerate(self._categories): for row, cat in enumerate(self._categories):
if table.rowCount() > 0: if cat.rowCount() > 0:
parent = self.index(row, 0) parent = self.index(row, 0)
index = self.index(0, 0, parent) index = self.index(0, 0, parent)
qtutils.ensure_valid(index) qtutils.ensure_valid(index)
@ -257,18 +224,11 @@ class SqlCompletionModel(QAbstractItemModel):
def last_item(self): def last_item(self):
"""Return the index of the last child (non-category) in the model.""" """Return the index of the last child (non-category) in the model."""
for row, table in reversed(list(enumerate(self._categories))): for row, cat in reversed(list(enumerate(self._categories))):
childcount = table.rowCount() childcount = cat.rowCount()
if childcount > 0: if childcount > 0:
parent = self.index(row, 0) parent = self.index(row, 0)
index = self.index(childcount - 1, 0, parent) index = self.index(childcount - 1, 0, parent)
qtutils.ensure_valid(index) qtutils.ensure_valid(index)
return index return index
return QModelIndex() return QModelIndex()
class SqlException(Exception):
"""Raised on an error interacting with the SQL database."""
pass

View File

@ -20,17 +20,16 @@
"""Functions that return config-related completion models.""" """Functions that return config-related completion models."""
from qutebrowser.config import configdata, configexc from qutebrowser.config import configdata, configexc
from qutebrowser.completion.models import base from qutebrowser.completion.models import completionmodel
from qutebrowser.utils import objreg from qutebrowser.utils import objreg
def section(): def section():
"""A CompletionModel filled with settings sections.""" """A CompletionModel filled with settings sections."""
model = base.CompletionModel(column_widths=(20, 70, 10)) model = completionmodel.CompletionModel(column_widths=(20, 70, 10))
cat = model.new_category("Sections") sections = ((name, configdata.SECTION_DESC[name].splitlines()[0].strip())
for name in configdata.DATA: for name in configdata.DATA)
desc = configdata.SECTION_DESC[name].splitlines()[0].strip() model.add_list("Sections", sections)
model.new_item(cat, name, desc)
return model return model
@ -40,12 +39,12 @@ def option(sectname):
Args: Args:
sectname: The name of the config section this model shows. sectname: The name of the config section this model shows.
""" """
model = base.CompletionModel(column_widths=(20, 70, 10)) model = completionmodel.CompletionModel(column_widths=(20, 70, 10))
cat = model.new_category(sectname)
try: try:
sectdata = configdata.DATA[sectname] sectdata = configdata.DATA[sectname]
except KeyError: except KeyError:
return None return None
options = []
for name in sectdata: for name in sectdata:
try: try:
desc = sectdata.descriptions[name] desc = sectdata.descriptions[name]
@ -57,7 +56,8 @@ def option(sectname):
desc = desc.splitlines()[0] desc = desc.splitlines()[0]
config = objreg.get('config') config = objreg.get('config')
val = config.get(sectname, name, raw=True) val = config.get(sectname, name, raw=True)
model.new_item(cat, name, desc, val) options.append((name, desc, val))
model.add_list(sectname, options)
return model return model
@ -68,16 +68,16 @@ def value(sectname, optname):
sectname: The name of the config section this model shows. sectname: The name of the config section this model shows.
optname: The name of the config option this model shows. optname: The name of the config option this model shows.
""" """
model = base.CompletionModel(column_widths=(20, 70, 10)) model = completionmodel.CompletionModel(column_widths=(20, 70, 10))
cur_cat = model.new_category("Current/Default")
config = objreg.get('config') config = objreg.get('config')
try: try:
val = config.get(sectname, optname, raw=True) or '""' current = config.get(sectname, optname, raw=True) or '""'
except (configexc.NoSectionError, configexc.NoOptionError): except (configexc.NoSectionError, configexc.NoOptionError):
return None return None
model.new_item(cur_cat, val, "Current value")
default_value = configdata.DATA[sectname][optname].default() or '""' default = configdata.DATA[sectname][optname].default() or '""'
model.new_item(cur_cat, default_value, "Default value")
if hasattr(configdata.DATA[sectname], 'valtype'): if hasattr(configdata.DATA[sectname], 'valtype'):
# Same type for all values (ValueList) # Same type for all values (ValueList)
vals = configdata.DATA[sectname].valtype.complete() vals = configdata.DATA[sectname].valtype.complete()
@ -87,8 +87,9 @@ def value(sectname, optname):
"sections, but {} is not!".format(sectname)) "sections, but {} is not!".format(sectname))
# Different type for each value (KeyValue) # Different type for each value (KeyValue)
vals = configdata.DATA[sectname][optname].typ.complete() vals = configdata.DATA[sectname][optname].typ.complete()
model.add_list("Current/Default", [(current, "Current value"),
(default, "Default value")])
if vals is not None: if vals is not None:
cat = model.new_category("Completions") model.add_list("Completions", vals)
for (val, desc) in vals:
model.new_item(cat, val, desc)
return model return model

View File

@ -0,0 +1,66 @@
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
# Copyright 2017 Ryan Roden-Corrent (rcorre) <ryan@rcorre.net>
#
# 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 <http://www.gnu.org/licenses/>.
"""The base completion model for completion in the command line.
Module attributes:
Role: An enum of user defined model roles.
"""
from PyQt5.QtCore import Qt
from PyQt5.QtGui import QStandardItem, QStandardItemModel
class ListCategory(QStandardItemModel):
"""Expose a list of items as a category for the CompletionModel."""
def __init__(self, name, items, parent=None):
super().__init__(parent)
self.name = name
# self.setColumnCount(3) TODO needed?
# TODO: batch insert?
# TODO: can I just insert a tuple instead of a list?
for item in items:
self.appendRow([QStandardItem(x) for x in item])
def flags(self, index):
"""Return the item flags for index.
Override QAbstractItemModel::flags.
Args:
index: The QModelIndex to get item flags for.
Return:
The item flags, or Qt.NoItemFlags on error.
"""
if not index.isValid():
return
if index.parent().isValid():
# item
return (Qt.ItemIsEnabled | Qt.ItemIsSelectable |
Qt.ItemNeverHasChildren)
else:
# category
return Qt.NoItemFlags
def set_pattern(self, pattern):
pass

View File

@ -22,30 +22,24 @@
from qutebrowser.config import config, configdata from qutebrowser.config import config, configdata
from qutebrowser.utils import objreg, log, qtutils from qutebrowser.utils import objreg, log, qtutils
from qutebrowser.commands import cmdutils from qutebrowser.commands import cmdutils
from qutebrowser.completion.models import base, sqlmodel from qutebrowser.completion.models import completionmodel
def command(): def command():
"""A CompletionModel filled with non-hidden commands and descriptions.""" """A CompletionModel filled with non-hidden commands and descriptions."""
model = base.CompletionModel(column_widths=(20, 60, 20)) model = completionmodel.CompletionModel(column_widths=(20, 60, 20))
cmdlist = _get_cmd_completions(include_aliases=True, include_hidden=False) cmdlist = _get_cmd_completions(include_aliases=True, include_hidden=False)
cat = model.new_category("Commands") model.add_list("Commands", cmdlist)
for (name, desc, misc) in cmdlist:
model.new_item(cat, name, desc, misc)
return model return model
def helptopic(): def helptopic():
"""A CompletionModel filled with help topics.""" """A CompletionModel filled with help topics."""
model = base.CompletionModel() model = completionmodel.CompletionModel()
cmdlist = _get_cmd_completions(include_aliases=False, include_hidden=True, cmdlist = _get_cmd_completions(include_aliases=False, include_hidden=True,
prefix=':') prefix=':')
cat = model.new_category("Commands") settings = []
for (name, desc, misc) in cmdlist:
model.new_item(cat, name, desc, misc)
cat = model.new_category("Settings")
for sectname, sectdata in configdata.DATA.items(): for sectname, sectdata in configdata.DATA.items():
for optname in sectdata: for optname in sectdata:
try: try:
@ -57,34 +51,35 @@ def helptopic():
else: else:
desc = desc.splitlines()[0] desc = desc.splitlines()[0]
name = '{}->{}'.format(sectname, optname) name = '{}->{}'.format(sectname, optname)
model.new_item(cat, name, desc) settings.append((name, desc))
model.add_list("Commands", cmdlist)
model.add_list("Settings", settings)
return model return model
def quickmark(): def quickmark():
"""A CompletionModel filled with all quickmarks.""" """A CompletionModel filled with all quickmarks."""
model = base.CompletionModel() model = completionmodel.CompletionModel(column_widths=(30, 70, 0))
model = sqlmodel.SqlCompletionModel(column_widths=(30, 70, 0)) model.add_sqltable('Quickmarks')
model.new_category('Quickmarks')
return model return model
def bookmark(): def bookmark():
"""A CompletionModel filled with all bookmarks.""" """A CompletionModel filled with all bookmarks."""
model = base.CompletionModel() model = completionmodel.CompletionModel(column_widths=(30, 70, 0))
model = sqlmodel.SqlCompletionModel(column_widths=(30, 70, 0)) model.add_sqltable('Bookmarks')
model.new_category('Bookmarks')
return model return model
def session(): def session():
"""A CompletionModel filled with session names.""" """A CompletionModel filled with session names."""
model = base.CompletionModel() model = completionmodel.CompletionModel()
cat = model.new_category("Sessions")
try: try:
for name in objreg.get('session-manager').list_sessions(): manager = objreg.get('session-manager')
if not name.startswith('_'): sessions = ((name,) for name in manager.list_sessions()
model.new_item(cat, name) if not name.startswith('_'))
model.add_list("Sessions", sessions)
except OSError: except OSError:
log.completion.exception("Failed to list sessions!") log.completion.exception("Failed to list sessions!")
return model return model
@ -111,7 +106,7 @@ def buffer():
window=int(win_id)) window=int(win_id))
tabbed_browser.on_tab_close_requested(int(tab_index) - 1) tabbed_browser.on_tab_close_requested(int(tab_index) - 1)
model = base.CompletionModel( model = completionmodel.CompletionModel(
column_widths=(6, 40, 54), column_widths=(6, 40, 54),
delete_cur_item=delete_buffer, delete_cur_item=delete_buffer,
columns_to_filter=[idx_column, url_column, text_column]) columns_to_filter=[idx_column, url_column, text_column])
@ -121,12 +116,13 @@ def buffer():
window=win_id) window=win_id)
if tabbed_browser.shutting_down: if tabbed_browser.shutting_down:
continue continue
c = model.new_category("{}".format(win_id)) tabs = []
for idx in range(tabbed_browser.count()): for idx in range(tabbed_browser.count()):
tab = tabbed_browser.widget(idx) tab = tabbed_browser.widget(idx)
model.new_item(c, "{}/{}".format(win_id, idx + 1), tabs.append(("{}/{}".format(win_id, idx + 1),
tab.url().toDisplayString(), tab.url().toDisplayString(),
tabbed_browser.page_title(idx)) tabbed_browser.page_title(idx)))
model.add_list("{}".format(win_id), tabs)
return model return model
@ -137,11 +133,9 @@ def bind(_key):
_key: the key being bound. _key: the key being bound.
""" """
# TODO: offer a 'Current binding' completion based on the key. # TODO: offer a 'Current binding' completion based on the key.
model = base.CompletionModel(column_widths=(20, 60, 20)) model = completionmodel.CompletionModel(column_widths=(20, 60, 20))
cmdlist = _get_cmd_completions(include_hidden=True, include_aliases=True) cmdlist = _get_cmd_completions(include_hidden=True, include_aliases=True)
cat = model.new_category("Commands") model.add_list("Commands", cmdlist)
for (name, desc, misc) in cmdlist:
model.new_item(cat, name, desc, misc)
return model return model

View File

@ -41,12 +41,14 @@ class CompletionFilterModel(QSortFilterProxyModel):
a long time for some reason. a long time for some reason.
""" """
def __init__(self, source, parent=None): def __init__(self, source, columns_to_filter, parent=None):
super().__init__(parent) super().__init__(parent)
super().setSourceModel(source) super().setSourceModel(source)
self.srcmodel = source self.srcmodel = source
self.pattern = '' self.pattern = ''
self.pattern_re = None self.pattern_re = None
self.columns_to_filter = columns_to_filter
self.name = source.name
def set_pattern(self, val): def set_pattern(self, val):
"""Setter for pattern. """Setter for pattern.
@ -64,41 +66,6 @@ class CompletionFilterModel(QSortFilterProxyModel):
sortcol = 0 sortcol = 0
self.sort(sortcol) self.sort(sortcol)
def count(self):
"""Get the count of non-toplevel items currently visible.
Note this only iterates one level deep, as we only need root items
(categories) and children (items) in our model.
"""
count = 0
for i in range(self.rowCount()):
cat = self.index(i, 0)
qtutils.ensure_valid(cat)
count += self.rowCount(cat)
return count
def first_item(self):
"""Return the first item in the model."""
for i in range(self.rowCount()):
cat = self.index(i, 0)
qtutils.ensure_valid(cat)
if cat.model().hasChildren(cat):
index = self.index(0, 0, cat)
qtutils.ensure_valid(index)
return index
return QModelIndex()
def last_item(self):
"""Return the last item in the model."""
for i in range(self.rowCount() - 1, -1, -1):
cat = self.index(i, 0)
qtutils.ensure_valid(cat)
if cat.model().hasChildren(cat):
index = self.index(self.rowCount(cat) - 1, 0, cat)
qtutils.ensure_valid(index)
return index
return QModelIndex()
def setSourceModel(self, model): def setSourceModel(self, model):
"""Override QSortFilterProxyModel's setSourceModel to clear pattern.""" """Override QSortFilterProxyModel's setSourceModel to clear pattern."""
log.completion.debug("Setting source model: {}".format(model)) log.completion.debug("Setting source model: {}".format(model))
@ -119,10 +86,10 @@ class CompletionFilterModel(QSortFilterProxyModel):
True if self.pattern is contained in item, or if it's a root item True if self.pattern is contained in item, or if it's a root item
(category). False in all other cases (category). False in all other cases
""" """
if parent == QModelIndex() or not self.pattern: if not self.pattern:
return True return True
for col in self.srcmodel.columns_to_filter: for col in self.columns_to_filter:
idx = self.srcmodel.index(row, col, parent) idx = self.srcmodel.index(row, col, parent)
if not idx.isValid(): # pragma: no cover if not idx.isValid(): # pragma: no cover
# this is a sanity check not hit by any test case # this is a sanity check not hit by any test case

View File

@ -0,0 +1,64 @@
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
# Copyright 2017 Ryan Roden-Corrent (rcorre) <ryan@rcorre.net>
#
# 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 <http://www.gnu.org/licenses/>.
"""A completion model backed by SQL tables."""
import re
from PyQt5.QtSql import QSqlQueryModel
from qutebrowser.misc import sql
class SqlCategory(QSqlQueryModel):
"""Wraps a SqlQuery for use as a completion category."""
def __init__(self, name, *, sort_by, sort_order, select, where,
columns_to_filter, parent=None):
super().__init__(parent=parent)
self.name = name
query = sql.run_query('select * from {} limit 1'.format(name))
self._fields = [query.record().fieldName(i) for i in columns_to_filter]
querystr = 'select {} from {} where ('.format(select, name)
# the incoming pattern will have literal % and _ escaped with '\'
# we need to tell sql to treat '\' as an escape character
querystr += ' or '.join("{} like ? escape '\\'".format(f)
for f in self._fields)
querystr += ')'
if where:
querystr += ' and ' + where
if sort_by:
assert sort_order == 'asc' or sort_order == 'desc'
querystr += ' order by {} {}'.format(sort_by, sort_order)
self._querystr = querystr
self.set_pattern('')
def set_pattern(self, pattern):
# escape to treat a user input % or _ as a literal, not a wildcard
pattern = pattern.replace('%', '\\%')
pattern = pattern.replace('_', '\\_')
# treat spaces as wildcards to match any of the typed words
pattern = re.sub(r' +', '%', pattern)
pattern = '%{}%'.format(pattern)
query = sql.run_query(self._querystr, [pattern for _ in self._fields])
self.setQuery(query)

View File

@ -19,7 +19,7 @@
"""Function to return the url completion model for the `open` command.""" """Function to return the url completion model for the `open` command."""
from qutebrowser.completion.models import sqlmodel from qutebrowser.completion.models import completionmodel
from qutebrowser.config import config from qutebrowser.config import config
from qutebrowser.utils import qtutils, log, objreg from qutebrowser.utils import qtutils, log, objreg
@ -61,14 +61,16 @@ def url():
Used for the `open` command. Used for the `open` command.
""" """
model = sqlmodel.SqlCompletionModel(column_widths=(40, 50, 10), model = completionmodel.CompletionModel(
column_widths=(40, 50, 10),
columns_to_filter=[_URLCOL, _TEXTCOL], columns_to_filter=[_URLCOL, _TEXTCOL],
delete_cur_item=_delete_url) delete_cur_item=_delete_url)
timefmt = config.get('completion', 'timestamp-format') timefmt = config.get('completion', 'timestamp-format')
select_time = "strftime('{}', atime, 'unixepoch')".format(timefmt) select_time = "strftime('{}', atime, 'unixepoch')".format(timefmt)
model.new_category('Quickmarks', select='url, name') model.add_sqltable('Quickmarks', select='url, name')
model.new_category('Bookmarks') model.add_sqltable('Bookmarks')
model.new_category('History', model.add_sqltable('History',
sort_order='desc', sort_by='atime', sort_order='desc', sort_by='atime',
select='url, title, {}'.format(select_time), select='url, title, {}'.format(select_time),
where='not redirect') where='not redirect')

View File

@ -0,0 +1,146 @@
# vim: ft=python fileencoding=utf-8 sts=4 sw=4 et:
# Copyright 2017 Ryan Roden-Corrent (rcorre) <ryan@rcorre.net>
#
# 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 <http://www.gnu.org/licenses/>.
"""Tests for CompletionModel."""
import pytest
from qutebrowser.completion.models import completionmodel, sortfilter
def _create_model(data, filter_cols=None):
"""Create a completion model populated with the given data.
data: A list of lists, where each sub-list represents a category, each
tuple in the sub-list represents an item, and each value in the
tuple represents the item data for that column
filter_cols: Columns to filter, or None for default.
"""
model = completionmodel.CompletionModel(columns_to_filter=filter_cols)
for catdata in data:
model.add_list('', catdata)
return model
def _extract_model_data(model):
"""Express a model's data as a list for easier comparison.
Return: A list of lists, where each sub-list represents a category, each
tuple in the sub-list represents an item, and each value in the
tuple represents the item data for that column
"""
data = []
for i in range(0, model.rowCount()):
cat_idx = model.index(i, 0)
row = []
for j in range(0, model.rowCount(cat_idx)):
row.append((model.data(cat_idx.child(j, 0)),
model.data(cat_idx.child(j, 1)),
model.data(cat_idx.child(j, 2))))
data.append(row)
return data
@pytest.mark.parametrize('tree, first, last', [
([[('Aa',)]], 'Aa', 'Aa'),
([[('Aa',)], [('Ba',)]], 'Aa', 'Ba'),
([[('Aa',), ('Ab',), ('Ac',)], [('Ba',), ('Bb',)], [('Ca',)]],
'Aa', 'Ca'),
([[], [('Ba',)]], 'Ba', 'Ba'),
([[], [], [('Ca',)]], 'Ca', 'Ca'),
([[], [], [('Ca',), ('Cb',)]], 'Ca', 'Cb'),
([[('Aa',)], []], 'Aa', 'Aa'),
([[('Aa',)], []], 'Aa', 'Aa'),
([[('Aa',)], [], []], 'Aa', 'Aa'),
([[('Aa',)], [], [('Ca',)]], 'Aa', 'Ca'),
([[], []], None, None),
])
def test_first_last_item(tree, first, last):
"""Test that first() and last() return indexes to the first and last items.
Args:
tree: Each list represents a completion category, with each string
being an item under that category.
first: text of the first item
last: text of the last item
"""
model = _create_model(tree)
assert model.data(model.first_item()) == first
assert model.data(model.last_item()) == last
@pytest.mark.parametrize('tree, expected', [
([[('Aa',)]], 1),
([[('Aa',)], [('Ba',)]], 2),
([[('Aa',), ('Ab',), ('Ac',)], [('Ba',), ('Bb',)], [('Ca',)]], 6),
([[], [('Ba',)]], 1),
([[], [], [('Ca',)]], 1),
([[], [], [('Ca',), ('Cb',)]], 2),
([[('Aa',)], []], 1),
([[('Aa',)], []], 1),
([[('Aa',)], [], []], 1),
([[('Aa',)], [], [('Ca',)]], 2),
])
def test_count(tree, expected):
model = _create_model(tree)
assert model.count() == expected
@pytest.mark.parametrize('pattern, filter_cols, before, after', [
('foo', [0],
[[('foo', '', ''), ('bar', '', '')]],
[[('foo', '', '')]]),
('foo', [0],
[[('foob', '', ''), ('fooc', '', ''), ('fooa', '', '')]],
[[('fooa', '', ''), ('foob', '', ''), ('fooc', '', '')]]),
('foo', [0],
[[('foo', '', '')], [('bar', '', '')]],
[[('foo', '', '')], []]),
# prefer foobar as it starts with the pattern
('foo', [0],
[[('barfoo', '', ''), ('foobar', '', '')]],
[[('foobar', '', ''), ('barfoo', '', '')]]),
# however, don't rearrange categories
('foo', [0],
[[('barfoo', '', '')], [('foobar', '', '')]],
[[('barfoo', '', '')], [('foobar', '', '')]]),
('foo', [1],
[[('foo', 'bar', ''), ('bar', 'foo', '')]],
[[('bar', 'foo', '')]]),
('foo', [0, 1],
[[('foo', 'bar', ''), ('bar', 'foo', ''), ('bar', 'bar', '')]],
[[('foo', 'bar', ''), ('bar', 'foo', '')]]),
('foo', [0, 1, 2],
[[('foo', '', ''), ('bar', '')]],
[[('foo', '', '')]]),
])
def test_set_pattern(pattern, filter_cols, before, after):
"""Validate the filtering and sorting results of set_pattern."""
# TODO: just test that it calls the mock on its child categories
model = _create_model(before, filter_cols)
model.set_pattern(pattern)
actual = _extract_model_data(model)
assert actual == after

View File

@ -25,7 +25,7 @@ import pytest
from PyQt5.QtGui import QStandardItem, QColor from PyQt5.QtGui import QStandardItem, QColor
from qutebrowser.completion import completionwidget from qutebrowser.completion import completionwidget
from qutebrowser.completion.models import base, sortfilter from qutebrowser.completion.models import completionmodel
from qutebrowser.commands import cmdexc from qutebrowser.commands import cmdexc
@ -72,28 +72,25 @@ def completionview(qtbot, status_command_stub, config_stub, win_registry,
@pytest.fixture @pytest.fixture
def simplemodel(completionview): def simplemodel(completionview):
"""A filter model wrapped around a completion model with one item.""" """A completion model with one item."""
model = base.CompletionModel() model = completionmodel.CompletionModel()
cat = QStandardItem() model.add_list('', [('foo'),])
cat.appendRow(QStandardItem('foo')) return model
model.appendRow(cat)
return sortfilter.CompletionFilterModel(model, parent=completionview)
def test_set_model(completionview): def test_set_model(completionview):
"""Ensure set_model actually sets the model and expands all categories.""" """Ensure set_model actually sets the model and expands all categories."""
model = base.CompletionModel() model = completionmodel.CompletionModel()
filtermodel = sortfilter.CompletionFilterModel(model)
for i in range(3): for i in range(3):
model.appendRow(QStandardItem(str(i))) model.add_list(str(i), [('foo',)])
completionview.set_model(filtermodel) completionview.set_model(model)
assert completionview.model() is filtermodel assert completionview.model() is model
for i in range(model.rowCount()): for i in range(model.rowCount()):
assert completionview.isExpanded(filtermodel.index(i, 0)) assert completionview.isExpanded(model.index(i, 0))
def test_set_pattern(completionview): def test_set_pattern(completionview):
model = sortfilter.CompletionFilterModel(base.CompletionModel()) model = completionmodel.CompletionModel()
model.set_pattern = unittest.mock.Mock() model.set_pattern = unittest.mock.Mock()
completionview.set_model(model, 'foo') completionview.set_model(model, 'foo')
model.set_pattern.assert_called_with('foo') model.set_pattern.assert_called_with('foo')
@ -159,15 +156,10 @@ def test_completion_item_focus(which, tree, expected, completionview, qtbot):
successive movement. None implies no signal should be successive movement. None implies no signal should be
emitted. emitted.
""" """
model = base.CompletionModel() model = completionmodel.CompletionModel()
for catdata in tree: for catdata in tree:
cat = QStandardItem() model.add_list('', (x,) for x in catdata)
model.appendRow(cat) completionview.set_model(model)
for name in catdata:
cat.appendRow(QStandardItem(name))
filtermodel = sortfilter.CompletionFilterModel(model,
parent=completionview)
completionview.set_model(filtermodel)
for entry in expected: for entry in expected:
if entry is None: if entry is None:
with qtbot.assertNotEmitted(completionview.selection_changed): with qtbot.assertNotEmitted(completionview.selection_changed):
@ -187,10 +179,8 @@ def test_completion_item_focus_no_model(which, completionview, qtbot):
""" """
with qtbot.assertNotEmitted(completionview.selection_changed): with qtbot.assertNotEmitted(completionview.selection_changed):
completionview.completion_item_focus(which) completionview.completion_item_focus(which)
model = base.CompletionModel() model = completionmodel.CompletionModel()
filtermodel = sortfilter.CompletionFilterModel(model, completionview.set_model(model)
parent=completionview)
completionview.set_model(filtermodel)
completionview.set_model(None) completionview.set_model(None)
with qtbot.assertNotEmitted(completionview.selection_changed): with qtbot.assertNotEmitted(completionview.selection_changed):
completionview.completion_item_focus(which) completionview.completion_item_focus(which)
@ -211,16 +201,12 @@ def test_completion_show(show, rows, quick_complete, completionview,
config_stub.data['completion']['show'] = show config_stub.data['completion']['show'] = show
config_stub.data['completion']['quick-complete'] = quick_complete config_stub.data['completion']['quick-complete'] = quick_complete
model = base.CompletionModel() model = completionmodel.CompletionModel()
for name in rows: for name in rows:
cat = QStandardItem() model.add_list('', [(name,)])
model.appendRow(cat)
cat.appendRow(QStandardItem(name))
filtermodel = sortfilter.CompletionFilterModel(model,
parent=completionview)
assert not completionview.isVisible() assert not completionview.isVisible()
completionview.set_model(filtermodel) completionview.set_model(model)
assert completionview.isVisible() == (show == 'always' and len(rows) > 0) assert completionview.isVisible() == (show == 'always' and len(rows) > 0)
completionview.completion_item_focus('next') completionview.completion_item_focus('next')
expected = (show != 'never' and len(rows) > 0 and expected = (show != 'never' and len(rows) > 0 and

View File

@ -228,12 +228,12 @@ def test_help_completion(qtmodeltester, monkeypatch, stubs, key_config_stub):
(':hide', '', ''), (':hide', '', ''),
], ],
"Settings": [ "Settings": [
('general->time', 'Is an illusion.', ''), ('general->time', 'Is an illusion.', None),
('general->volume', 'Goes to 11', ''), ('general->volume', 'Goes to 11', None),
('ui->gesture', 'Waggle your hands to control qutebrowser', ''), ('ui->gesture', 'Waggle your hands to control qutebrowser', None),
('ui->mind', 'Enable mind-control ui (experimental)', ''), ('ui->mind', 'Enable mind-control ui (experimental)', None),
('ui->voice', 'Whether to respond to voice commands', ''), ('ui->voice', 'Whether to respond to voice commands', None),
('searchengines->DEFAULT', '', ''), ('searchengines->DEFAULT', '', None),
] ]
}) })
@ -342,7 +342,9 @@ def test_session_completion(qtmodeltester, session_manager_stub):
qtmodeltester.check(model) qtmodeltester.check(model)
_check_completions(model, { _check_completions(model, {
"Sessions": [('default', '', ''), ('1', '', ''), ('2', '', '')] "Sessions": [('default', None, None),
('1', None, None),
('2', None, None)]
}) })
@ -406,9 +408,9 @@ def test_setting_section_completion(qtmodeltester, monkeypatch, stubs):
_check_completions(model, { _check_completions(model, {
"Sections": [ "Sections": [
('general', 'General/miscellaneous options.', ''), ('general', 'General/miscellaneous options.', None),
('ui', 'General options related to the user interface.', ''), ('ui', 'General options related to the user interface.', None),
('searchengines', 'Definitions of search engines ...', ''), ('searchengines', 'Definitions of search engines ...', None),
] ]
}) })
@ -462,12 +464,12 @@ def test_setting_value_completion(qtmodeltester, monkeypatch, stubs,
_check_completions(model, { _check_completions(model, {
"Current/Default": [ "Current/Default": [
('0', 'Current value', ''), ('0', 'Current value', None),
('11', 'Default value', ''), ('11', 'Default value', None),
], ],
"Completions": [ "Completions": [
('0', '', ''), ('0', '', None),
('11', '', ''), ('11', '', None),
] ]
}) })

View File

@ -21,9 +21,10 @@
import pytest import pytest
from qutebrowser.completion.models import base, sortfilter from qutebrowser.completion.models import listcategory, sortfilter
# TODO: merge listcategory and sortfilter
def _create_model(data): def _create_model(data):
"""Create a completion model populated with the given data. """Create a completion model populated with the given data.
@ -31,11 +32,9 @@ def _create_model(data):
tuple in the sub-list represents an item, and each value in the tuple in the sub-list represents an item, and each value in the
tuple represents the item data for that column tuple represents the item data for that column
""" """
model = base.CompletionModel() model = completionmodel.CompletionModel()
for catdata in data: for catdata in data:
cat = model.new_category('') cat = model.add_list(itemdata)
for itemdata in catdata:
model.new_item(cat, *itemdata)
return model return model
@ -72,7 +71,7 @@ def _extract_model_data(model):
('4', 'blah', False), ('4', 'blah', False),
]) ])
def test_filter_accepts_row(pattern, data, expected): def test_filter_accepts_row(pattern, data, expected):
source_model = base.CompletionModel() source_model = completionmodel.CompletionModel()
cat = source_model.new_category('test') cat = source_model.new_category('test')
source_model.new_item(cat, data) source_model.new_item(cat, data)
@ -86,35 +85,6 @@ def test_filter_accepts_row(pattern, data, expected):
assert row_count == (1 if expected else 0) assert row_count == (1 if expected else 0)
@pytest.mark.parametrize('tree, first, last', [
([[('Aa',)]], 'Aa', 'Aa'),
([[('Aa',)], [('Ba',)]], 'Aa', 'Ba'),
([[('Aa',), ('Ab',), ('Ac',)], [('Ba',), ('Bb',)], [('Ca',)]],
'Aa', 'Ca'),
([[], [('Ba',)]], 'Ba', 'Ba'),
([[], [], [('Ca',)]], 'Ca', 'Ca'),
([[], [], [('Ca',), ('Cb',)]], 'Ca', 'Cb'),
([[('Aa',)], []], 'Aa', 'Aa'),
([[('Aa',)], []], 'Aa', 'Aa'),
([[('Aa',)], [], []], 'Aa', 'Aa'),
([[('Aa',)], [], [('Ca',)]], 'Aa', 'Ca'),
([[], []], None, None),
])
def test_first_last_item(tree, first, last):
"""Test that first() and last() return indexes to the first and last items.
Args:
tree: Each list represents a completion category, with each string
being an item under that category.
first: text of the first item
last: text of the last item
"""
model = _create_model(tree)
filter_model = sortfilter.CompletionFilterModel(model)
assert filter_model.data(filter_model.first_item()) == first
assert filter_model.data(filter_model.last_item()) == last
def test_set_source_model(): def test_set_source_model():
"""Ensure setSourceModel sets source_model and clears the pattern.""" """Ensure setSourceModel sets source_model and clears the pattern."""
model1 = base.CompletionModel() model1 = base.CompletionModel()
@ -131,24 +101,6 @@ def test_set_source_model():
assert not filter_model.pattern assert not filter_model.pattern
@pytest.mark.parametrize('tree, expected', [
([[('Aa',)]], 1),
([[('Aa',)], [('Ba',)]], 2),
([[('Aa',), ('Ab',), ('Ac',)], [('Ba',), ('Bb',)], [('Ca',)]], 6),
([[], [('Ba',)]], 1),
([[], [], [('Ca',)]], 1),
([[], [], [('Ca',), ('Cb',)]], 2),
([[('Aa',)], []], 1),
([[('Aa',)], []], 1),
([[('Aa',)], [], []], 1),
([[('Aa',)], [], [('Ca',)]], 2),
])
def test_count(tree, expected):
model = _create_model(tree)
filter_model = sortfilter.CompletionFilterModel(model)
assert filter_model.count() == expected
@pytest.mark.parametrize('pattern, filter_cols, before, after', [ @pytest.mark.parametrize('pattern, filter_cols, before, after', [
('foo', [0], ('foo', [0],
[[('foo', '', ''), ('bar', '', '')]], [[('foo', '', ''), ('bar', '', '')]],