2014-01-29 09:06:38 +01:00
|
|
|
"""Completion view which appears when something is typed in the statusbar
|
|
|
|
command section.
|
|
|
|
|
|
|
|
Defines a CompletionView which uses CompletionFiterModel and CompletionModel
|
|
|
|
subclasses to provide completions.
|
|
|
|
"""
|
|
|
|
|
2014-01-27 21:35:12 +01:00
|
|
|
import html
|
|
|
|
|
|
|
|
from PyQt5.QtWidgets import (QTreeView, QStyledItemDelegate, QStyle,
|
2014-01-30 22:29:01 +01:00
|
|
|
QStyleOptionViewItem, QSizePolicy)
|
2014-01-27 21:35:12 +01:00
|
|
|
from PyQt5.QtCore import (QRectF, QRect, QPoint, pyqtSignal, Qt,
|
|
|
|
QItemSelectionModel)
|
|
|
|
from PyQt5.QtGui import (QIcon, QPalette, QTextDocument, QTextOption,
|
|
|
|
QTextCursor)
|
|
|
|
|
2014-01-28 12:21:00 +01:00
|
|
|
import qutebrowser.utils.config as config
|
2014-01-27 21:35:12 +01:00
|
|
|
from qutebrowser.utils.completion import CompletionFilterModel
|
|
|
|
from qutebrowser.commands.utils import CommandCompletionModel
|
|
|
|
|
2014-01-28 23:04:02 +01:00
|
|
|
|
2014-01-27 21:35:12 +01:00
|
|
|
class CompletionView(QTreeView):
|
2014-01-29 09:06:38 +01:00
|
|
|
"""The view showing available completions.
|
|
|
|
|
|
|
|
Based on QTreeView but heavily customized so root elements show as category
|
|
|
|
headers, and children show as flat list.
|
|
|
|
|
|
|
|
Highlights completions based on marks in the UserRole.
|
|
|
|
"""
|
|
|
|
|
2014-01-27 21:35:12 +01:00
|
|
|
_stylesheet = """
|
2014-01-28 12:21:00 +01:00
|
|
|
QTreeView {{
|
2014-01-28 15:20:58 +01:00
|
|
|
font-family: {monospace};
|
2014-01-28 12:21:00 +01:00
|
|
|
{color[completion.fg]}
|
|
|
|
{color[completion.bg]}
|
2014-01-27 21:35:12 +01:00
|
|
|
outline: 0;
|
2014-01-28 12:21:00 +01:00
|
|
|
}}
|
|
|
|
QTreeView::item {{
|
|
|
|
{color[completion.item.fg]}
|
|
|
|
{color[completion.item.bg]}
|
|
|
|
}}
|
|
|
|
QTreeView::item:has-children {{
|
2014-01-27 21:35:12 +01:00
|
|
|
font-weight: bold;
|
2014-01-28 12:21:00 +01:00
|
|
|
{color[completion.category.fg]}
|
|
|
|
{color[completion.category.bg]}
|
|
|
|
border-top: 1px solid {color[completion.category.border.top]};
|
2014-01-28 17:33:48 +01:00
|
|
|
border-bottom: 1px solid
|
|
|
|
{color[completion.category.border.bottom]};
|
2014-01-28 12:21:00 +01:00
|
|
|
}}
|
|
|
|
QTreeView::item:selected {{
|
2014-01-28 13:03:15 +01:00
|
|
|
border-top: 1px solid {color[completion.item.selected.border.top]};
|
2014-01-28 17:33:48 +01:00
|
|
|
border-bottom: 1px solid
|
|
|
|
{color[completion.item.selected.border.bottom]};
|
2014-01-28 17:30:37 +01:00
|
|
|
{color[completion.item.selected.bg]}
|
2014-01-28 12:21:00 +01:00
|
|
|
{color[completion.item.selected.fg]}
|
|
|
|
}}
|
2014-01-27 21:35:12 +01:00
|
|
|
"""
|
|
|
|
# FIXME because we use :has-children, if a category is empty, it won't look
|
|
|
|
# like one anymore
|
|
|
|
# FIXME somehow only the first column is yellow, even with
|
|
|
|
# setAllColumnsShowFocus
|
|
|
|
completion_models = {}
|
|
|
|
append_cmd_text = pyqtSignal(str)
|
|
|
|
ignore_next = False
|
2014-01-28 12:59:12 +01:00
|
|
|
enabled = True
|
|
|
|
completing = False
|
2014-01-27 21:35:12 +01:00
|
|
|
|
|
|
|
def __init__(self, parent=None):
|
|
|
|
super().__init__(parent)
|
2014-01-28 12:59:12 +01:00
|
|
|
self.enabled = config.config.getboolean('general', 'show_completion',
|
|
|
|
fallback=True)
|
2014-01-27 21:35:12 +01:00
|
|
|
self.completion_models[''] = None
|
|
|
|
self.completion_models['command'] = CommandCompletionModel()
|
|
|
|
self.model = CompletionFilterModel()
|
|
|
|
self.setModel(self.model)
|
|
|
|
self.model.setSourceModel(self.completion_models['command'])
|
2014-01-27 22:18:12 +01:00
|
|
|
self.model.pattern_changed.connect(self.resort)
|
2014-01-27 21:35:12 +01:00
|
|
|
self.setItemDelegate(CompletionItemDelegate())
|
2014-01-28 19:52:09 +01:00
|
|
|
self.setStyleSheet(config.get_stylesheet(self._stylesheet))
|
2014-01-30 22:29:01 +01:00
|
|
|
self.setSizePolicy(QSizePolicy.Fixed, QSizePolicy.Minimum)
|
2014-01-27 21:35:12 +01:00
|
|
|
self.expandAll()
|
|
|
|
self.setHeaderHidden(True)
|
|
|
|
self.setIndentation(0)
|
|
|
|
self.setItemsExpandable(False)
|
|
|
|
self.hide()
|
|
|
|
# FIXME set elidemode
|
|
|
|
|
|
|
|
def resizeEvent(self, e):
|
2014-01-29 09:06:38 +01:00
|
|
|
"""Extends resizeEvent of QTreeView.
|
|
|
|
|
|
|
|
Always adjusts the column width to the new window width.
|
|
|
|
|
|
|
|
e -- The QResizeEvent.
|
|
|
|
"""
|
2014-01-27 21:35:12 +01:00
|
|
|
width = e.size().width()
|
2014-01-29 09:07:04 +01:00
|
|
|
cols = self.model.columnCount()
|
2014-01-29 15:40:55 +01:00
|
|
|
assert cols >= 1
|
|
|
|
assert width / cols > 1
|
2014-01-29 09:07:04 +01:00
|
|
|
for i in range(cols):
|
|
|
|
self.setColumnWidth(i, width / cols)
|
2014-01-27 21:35:12 +01:00
|
|
|
super().resizeEvent(e)
|
|
|
|
|
|
|
|
def setmodel(self, model):
|
2014-01-29 09:06:38 +01:00
|
|
|
"""Switch completion to a new model.
|
|
|
|
|
|
|
|
Called from cmd_text_changed().
|
|
|
|
|
|
|
|
model -- A QAbstractItemModel with available completions.
|
|
|
|
"""
|
2014-01-27 21:35:12 +01:00
|
|
|
self.model.setSourceModel(self.completion_models[model])
|
|
|
|
self.model.pattern = ''
|
|
|
|
self.expandAll()
|
|
|
|
|
2014-01-29 08:36:44 +01:00
|
|
|
def resort(self, pattern): # pylint: disable=unused-argument
|
2014-01-29 09:06:38 +01:00
|
|
|
"""Sort the available completions.
|
|
|
|
|
|
|
|
If the current completion model overrides sort(), it is used.
|
|
|
|
If not, the default implementation in QCompletionFilterModel is called.
|
|
|
|
"""
|
2014-01-29 09:07:04 +01:00
|
|
|
sortcol = 0
|
2014-01-27 22:18:12 +01:00
|
|
|
try:
|
2014-01-29 09:07:04 +01:00
|
|
|
self.model.sourceModel().sort(sortcol)
|
2014-01-27 22:18:12 +01:00
|
|
|
except NotImplementedError:
|
2014-01-29 09:07:04 +01:00
|
|
|
self.model.sort(sortcol)
|
2014-01-27 22:18:12 +01:00
|
|
|
|
2014-01-27 21:35:12 +01:00
|
|
|
def resize_to_bar(self, geom):
|
2014-01-29 09:06:38 +01:00
|
|
|
"""Resize the completion area to the statusbar geometry.
|
|
|
|
|
|
|
|
Slot for the resized signal of the statusbar.
|
|
|
|
geom -- A QRect containing the statusbar geometry.
|
|
|
|
"""
|
2014-01-27 21:35:12 +01:00
|
|
|
bottomleft = geom.topLeft()
|
|
|
|
bottomright = geom.topRight()
|
|
|
|
delta = QPoint(0, 200)
|
|
|
|
topleft = bottomleft - delta
|
2014-01-29 15:40:55 +01:00
|
|
|
assert topleft.x() < bottomright.x()
|
|
|
|
assert topleft.y() < bottomright.y()
|
2014-01-27 21:35:12 +01:00
|
|
|
self.setGeometry(QRect(topleft, bottomright))
|
|
|
|
|
|
|
|
def cmd_text_changed(self, text):
|
2014-01-29 09:06:38 +01:00
|
|
|
"""Check if completions are available and activate them.
|
|
|
|
|
|
|
|
Slot for the textChanged signal of the statusbar command widget.
|
|
|
|
text -- The new text
|
|
|
|
"""
|
2014-01-27 21:35:12 +01:00
|
|
|
if self.ignore_next:
|
2014-01-29 09:06:38 +01:00
|
|
|
# Text changed by a completion, so we don't have to complete again.
|
2014-01-27 21:35:12 +01:00
|
|
|
self.ignore_next = False
|
|
|
|
return
|
|
|
|
# FIXME more sophisticated completions
|
|
|
|
if ' ' in text or not text.startswith(':'):
|
|
|
|
self.hide()
|
2014-01-28 12:59:12 +01:00
|
|
|
self.completing = False
|
2014-01-27 21:35:12 +01:00
|
|
|
return
|
|
|
|
|
2014-01-28 12:59:12 +01:00
|
|
|
self.completing = True
|
2014-01-27 21:35:12 +01:00
|
|
|
self.setmodel('command')
|
|
|
|
text = text.lstrip(':')
|
|
|
|
self.model.pattern = text
|
2014-01-29 09:14:37 +01:00
|
|
|
self.model.sourceModel().mark_all_items(text)
|
2014-01-28 12:59:12 +01:00
|
|
|
if self.enabled:
|
|
|
|
self.show()
|
2014-01-27 21:35:12 +01:00
|
|
|
|
|
|
|
def tab_handler(self, shift):
|
2014-01-29 15:30:19 +01:00
|
|
|
"""Handle a tab press for the CompletionView.
|
|
|
|
|
|
|
|
Selects the previous/next item and writes the new text to the
|
|
|
|
statusbar. Called by key_(s)tab_handler in statusbar.command.
|
|
|
|
|
|
|
|
shift -- Whether shift is pressed or not.
|
|
|
|
"""
|
2014-01-28 12:59:12 +01:00
|
|
|
if not self.completing:
|
2014-01-29 15:30:19 +01:00
|
|
|
# No completion running at the moment, ignore keypress
|
2014-01-27 21:35:12 +01:00
|
|
|
return
|
2014-01-28 08:16:31 +01:00
|
|
|
idx = self._next_idx(shift)
|
|
|
|
self.selectionModel().setCurrentIndex(
|
|
|
|
idx, QItemSelectionModel.ClearAndSelect)
|
2014-01-28 13:23:08 +01:00
|
|
|
data = self.model.data(idx)
|
|
|
|
if data is not None:
|
2014-01-29 15:30:19 +01:00
|
|
|
self.ignore_next = True
|
2014-01-28 13:23:08 +01:00
|
|
|
self.append_cmd_text.emit(self.model.data(idx) + ' ')
|
2014-01-27 21:35:12 +01:00
|
|
|
|
2014-01-29 15:30:19 +01:00
|
|
|
def _next_idx(self, upwards):
|
|
|
|
"""Get the previous/next QModelIndex displayed in the view.
|
|
|
|
|
|
|
|
Used by tab_handler.
|
|
|
|
|
|
|
|
upwards -- Get previous item, not next.
|
|
|
|
"""
|
2014-01-28 08:16:31 +01:00
|
|
|
idx = self.selectionModel().currentIndex()
|
|
|
|
if not idx.isValid():
|
|
|
|
# No item selected yet
|
2014-01-29 09:14:37 +01:00
|
|
|
return self.model.first_item()
|
2014-01-28 08:16:31 +01:00
|
|
|
while True:
|
2014-01-29 15:30:19 +01:00
|
|
|
idx = self.indexAbove(idx) if upwards else self.indexBelow(idx)
|
2014-01-29 09:14:37 +01:00
|
|
|
# wrap around if we arrived at beginning/end
|
2014-01-29 15:30:19 +01:00
|
|
|
if not idx.isValid() and upwards:
|
2014-01-29 09:14:37 +01:00
|
|
|
return self.model.last_item()
|
2014-01-29 15:30:19 +01:00
|
|
|
elif not idx.isValid() and not upwards:
|
2014-01-29 09:14:37 +01:00
|
|
|
return self.model.first_item()
|
|
|
|
elif idx.parent().isValid():
|
2014-01-28 08:16:31 +01:00
|
|
|
# Item is a real item, not a category header -> success
|
|
|
|
return idx
|
|
|
|
|
2014-01-28 23:04:02 +01:00
|
|
|
|
2014-01-27 21:35:12 +01:00
|
|
|
class CompletionItemDelegate(QStyledItemDelegate):
|
2014-01-29 15:30:19 +01:00
|
|
|
"""Delegate used by CompletionView to draw individual items.
|
|
|
|
|
|
|
|
Mainly a cleaned up port of Qt's way to draw a TreeView item, except it
|
|
|
|
uses a QTextDocument to draw the text and add marking.
|
|
|
|
|
|
|
|
Original implementation:
|
|
|
|
qt/src/gui/styles/qcommonstyle.cpp:drawControl:2153
|
|
|
|
"""
|
|
|
|
|
2014-01-27 21:35:12 +01:00
|
|
|
opt = None
|
|
|
|
style = None
|
|
|
|
painter = None
|
|
|
|
|
|
|
|
def paint(self, painter, option, index):
|
2014-01-29 15:30:19 +01:00
|
|
|
"""Overrides the QStyledItemDelegate paint function."""
|
2014-01-27 21:35:12 +01:00
|
|
|
painter.save()
|
|
|
|
|
|
|
|
self.painter = painter
|
|
|
|
self.opt = QStyleOptionViewItem(option)
|
|
|
|
self.initStyleOption(self.opt, index)
|
|
|
|
self.style = self.opt.widget.style()
|
|
|
|
|
|
|
|
self._draw_background()
|
|
|
|
self._draw_icon()
|
|
|
|
self._draw_text(index)
|
|
|
|
self._draw_focus_rect()
|
|
|
|
|
|
|
|
painter.restore()
|
|
|
|
|
|
|
|
def _draw_background(self):
|
2014-01-29 15:30:19 +01:00
|
|
|
"""Draw the background of an ItemViewItem"""
|
2014-01-27 21:35:12 +01:00
|
|
|
self.style.drawPrimitive(self.style.PE_PanelItemViewItem, self.opt,
|
|
|
|
self.painter, self.opt.widget)
|
|
|
|
|
|
|
|
def _draw_icon(self):
|
2014-01-29 15:30:19 +01:00
|
|
|
"""Draw the icon of an ItemViewItem"""
|
2014-01-27 21:35:12 +01:00
|
|
|
icon_rect = self.style.subElementRect(
|
|
|
|
self.style.SE_ItemViewItemDecoration, self.opt, self.opt.widget)
|
|
|
|
|
|
|
|
mode = QIcon.Normal
|
|
|
|
if not self.opt.state & QStyle.State_Enabled:
|
|
|
|
mode = QIcon.Disabled
|
|
|
|
elif self.opt.state & QStyle.State_Selected:
|
|
|
|
mode = QIcon.Selected
|
|
|
|
state = QIcon.On if self.opt.state & QStyle.State_Open else QIcon.Off
|
|
|
|
self.opt.icon.paint(self.painter, icon_rect,
|
|
|
|
self.opt.decorationAlignment, mode, state)
|
|
|
|
|
|
|
|
def _draw_text(self, index):
|
2014-01-29 15:30:19 +01:00
|
|
|
"""Draw the text of an ItemViewItem.
|
|
|
|
|
|
|
|
This is the main part where we differ from the original implementation
|
|
|
|
in Qt: We use a QTextDocument to draw text.
|
|
|
|
|
|
|
|
index -- The QModelIndex of the item to draw.
|
|
|
|
"""
|
2014-01-27 21:35:12 +01:00
|
|
|
if not self.opt.text:
|
|
|
|
return
|
|
|
|
|
|
|
|
text_rect_ = self.style.subElementRect(self.style.SE_ItemViewItemText,
|
|
|
|
self.opt, self.opt.widget)
|
|
|
|
margin = self.style.pixelMetric(QStyle.PM_FocusFrameHMargin, self.opt,
|
|
|
|
self.opt.widget) + 1
|
|
|
|
# remove width padding
|
|
|
|
text_rect = text_rect_.adjusted(margin, 0, -margin, 0)
|
|
|
|
self.painter.save()
|
|
|
|
state = self.opt.state
|
2014-01-29 08:36:44 +01:00
|
|
|
if state & QStyle.State_Enabled and state & QStyle.State_Active:
|
2014-01-27 21:35:12 +01:00
|
|
|
cg = QPalette.Normal
|
|
|
|
elif state & QStyle.State_Enabled:
|
|
|
|
cg = QPalette.Inactive
|
|
|
|
else:
|
|
|
|
cg = QPalette.Disabled
|
|
|
|
|
|
|
|
if state & QStyle.State_Selected:
|
|
|
|
self.painter.setPen(self.opt.palette.color(
|
|
|
|
cg, QPalette.HighlightedText))
|
|
|
|
# FIXME this is a dirty fix for the text jumping by one pixel...
|
|
|
|
# we really should do this properly somehow
|
|
|
|
text_rect.adjust(0, -1, 0, 0)
|
|
|
|
else:
|
|
|
|
self.painter.setPen(self.opt.palette.color(cg, QPalette.Text))
|
|
|
|
|
|
|
|
if state & QStyle.State_Editing:
|
|
|
|
self.painter.setPen(self.opt.palette.color(cg, QPalette.Text))
|
|
|
|
self.painter.drawRect(text_rect_.adjusted(0, 0, -1, -1))
|
|
|
|
|
|
|
|
self.painter.translate(text_rect.left(), text_rect.top())
|
2014-01-29 06:36:13 +01:00
|
|
|
self._draw_textdoc(index, text_rect)
|
|
|
|
self.painter.restore()
|
|
|
|
|
|
|
|
def _draw_textdoc(self, index, text_rect):
|
2014-01-29 15:30:19 +01:00
|
|
|
"""Draw the QTextDocument of an item.
|
|
|
|
|
|
|
|
index -- The QModelIndex of the item to draw.
|
|
|
|
text_rect -- The QRect to clip the drawing to.
|
|
|
|
"""
|
2014-01-29 06:36:13 +01:00
|
|
|
# FIXME we probably should do eliding here. See
|
|
|
|
# qcommonstyle.cpp:viewItemDrawText
|
2014-01-27 21:35:12 +01:00
|
|
|
clip = QRectF(0, 0, text_rect.width(), text_rect.height())
|
|
|
|
|
|
|
|
text_option = QTextOption()
|
|
|
|
if self.opt.features & QStyleOptionViewItem.WrapText:
|
|
|
|
text_option.setWrapMode(QTextOption.WordWrap)
|
|
|
|
else:
|
|
|
|
text_option.setWrapMode(QTextOption.ManualWrap)
|
|
|
|
text_option.setTextDirection(self.opt.direction)
|
|
|
|
text_option.setAlignment(QStyle.visualAlignment(
|
|
|
|
self.opt.direction, self.opt.displayAlignment))
|
|
|
|
|
|
|
|
doc = QTextDocument()
|
|
|
|
if index.parent().isValid():
|
|
|
|
doc.setPlainText(self.opt.text)
|
|
|
|
else:
|
|
|
|
doc.setHtml('<b>{}</b>'.format(html.escape(self.opt.text)))
|
|
|
|
doc.setDefaultFont(self.opt.font)
|
|
|
|
doc.setDefaultTextOption(text_option)
|
2014-01-28 12:21:00 +01:00
|
|
|
doc.setDefaultStyleSheet(config.get_stylesheet("""
|
|
|
|
.highlight {{
|
|
|
|
{color[completion.match.fg]}
|
|
|
|
}}
|
|
|
|
"""))
|
2014-01-27 21:35:12 +01:00
|
|
|
doc.setDocumentMargin(0)
|
|
|
|
|
2014-01-28 07:53:05 +01:00
|
|
|
if index.column() == 0:
|
|
|
|
marks = index.data(Qt.UserRole)
|
|
|
|
for mark in marks:
|
|
|
|
cur = QTextCursor(doc)
|
|
|
|
cur.setPosition(mark[0])
|
|
|
|
cur.setPosition(mark[1], QTextCursor.KeepAnchor)
|
|
|
|
txt = cur.selectedText()
|
|
|
|
cur.removeSelectedText()
|
|
|
|
cur.insertHtml('<span class="highlight">{}</span>'.format(
|
|
|
|
html.escape(txt)))
|
2014-01-27 21:35:12 +01:00
|
|
|
doc.drawContents(self.painter, clip)
|
|
|
|
|
|
|
|
def _draw_focus_rect(self):
|
2014-01-29 15:30:19 +01:00
|
|
|
"""Draws the focus rectangle of an ItemViewItem"""
|
2014-01-27 21:35:12 +01:00
|
|
|
state = self.opt.state
|
|
|
|
if not state & QStyle.State_HasFocus:
|
|
|
|
return
|
|
|
|
o = self.opt
|
|
|
|
o.rect = self.style.subElementRect(self.style.SE_ItemViewItemFocusRect,
|
|
|
|
self.opt, self.opt.widget)
|
|
|
|
o.state |= QStyle.State_KeyboardFocusChange | QStyle.State_Item
|
|
|
|
if state & QStyle.State_Enabled:
|
|
|
|
cg = QPalette.Normal
|
|
|
|
else:
|
|
|
|
cg = QPalette.Disabled
|
|
|
|
if state & QStyle.State_Selected:
|
|
|
|
role = QPalette.Highlight
|
|
|
|
else:
|
|
|
|
role = QPalette.Window
|
|
|
|
o.backgroundColor = self.opt.palette.color(cg, role)
|
|
|
|
self.style.drawPrimitive(QStyle.PE_FrameFocusRect, o, self.painter,
|
|
|
|
self.opt.widget)
|