artiq/artiq/gui/fuzzy_select.py
David Nadlinger 69718fca90 gui: Improve fuzzy-select heuristics
Even though the code already used non-greedy wildcards before,
it would not find the shortest match, as earlier match starts
would still take precedence.

This could possibly be sped up a bit in CPython by doing
everything inside re using lookahead-assertion trickery, but the
current code is already imperceptibly fast for hundreds of
choices.
2020-08-14 02:13:45 +01:00

295 lines
11 KiB
Python

import re
from functools import partial
from typing import List, Tuple
from PyQt5 import QtCore, QtWidgets
from artiq.gui.tools import LayoutWidget
class FuzzySelectWidget(LayoutWidget):
"""Widget to select from a list of pre-defined choices by typing in a
substring match (cf. Ctrl+P "Quick Open"/"Goto anything" functions in
editors/IDEs).
"""
#: Raised when the selection process is aborted by the user (Esc, loss of
#: focus, etc.).
aborted = QtCore.pyqtSignal()
#: Raised when an entry has been selected, giving the label of the user
#: choice and any additional QEvent.modifiers() (e.g. Ctrl key pressed).
finished = QtCore.pyqtSignal(str, int)
def __init__(self,
choices: List[Tuple[str, int]] = [],
entry_count_limit: int = 10,
*args):
"""
:param choices: The choices the user can select from, given as tuples
of labels to display and an additional weight added to the
fuzzy-matching score.
:param entry_count_limit: Maximum number of entries to show.
"""
super().__init__(*args)
self.entry_count_limit = entry_count_limit
assert entry_count_limit >= 2, ("Need to allow at least two entries " +
"to show the '<n> not shown' hint")
self.line_edit = QtWidgets.QLineEdit(self)
self.layout.addWidget(self.line_edit)
line_edit_focus_filter = _FocusEventFilter(self.line_edit)
line_edit_focus_filter.focus_gained.connect(self._activate)
line_edit_focus_filter.focus_lost.connect(self._line_edit_focus_lost)
self.line_edit.installEventFilter(line_edit_focus_filter)
self.line_edit.textChanged.connect(self._update_menu)
escape_filter = _EscapeKeyFilter(self)
escape_filter.escape_pressed.connect(self.abort)
self.line_edit.installEventFilter(escape_filter)
self.menu = None
self.update_when_text_changed = True
self.menu_typing_filter = None
self.line_edit_up_down_filter = None
self.abort_when_menu_hidden = False
self.abort_when_line_edit_unfocussed = True
self.set_choices(choices)
def set_choices(self, choices: List[Tuple[str, int]]) -> None:
"""Update the list of choices available to the user."""
# Keep sorted in the right order for when the query is empty.
self.choices = sorted(choices, key=lambda a: (a[1], a[0]))
if self.menu:
self._update_menu()
def _activate(self):
self.update_when_text_changed = True
if not self.menu:
# Show menu after initial layout is complete.
QtCore.QTimer.singleShot(0, self._update_menu)
def _ensure_menu(self):
if self.menu:
return
self.menu = QtWidgets.QMenu(self)
# Display menu with search results beneath line edit.
menu_pos = self.line_edit.mapToGlobal(self.line_edit.pos())
menu_pos.setY(menu_pos.y() + self.line_edit.height())
self.menu.popup(menu_pos)
self.menu.aboutToHide.connect(self._menu_hidden)
def _menu_hidden(self):
if self.abort_when_menu_hidden:
self.abort_when_menu_hidden = False
self.abort()
def _line_edit_focus_lost(self):
if self.abort_when_line_edit_unfocussed:
self.abort()
def _update_menu(self):
if not self.update_when_text_changed:
return
filtered_choices = self._filter_choices()
if not filtered_choices:
# No matches, don't display menu at all.
if self.menu:
self.abort_when_menu_hidden = False
self.menu.close()
self.menu = None
self.abort_when_line_edit_unfocussed = True
self.line_edit.setFocus()
return
# Truncate the list, leaving room for the "<n> not shown" entry.
num_omitted = 0
if len(filtered_choices) > self.entry_count_limit:
num_omitted = len(filtered_choices) - (self.entry_count_limit - 1)
filtered_choices = filtered_choices[:self.entry_count_limit - 1]
# We are going to end up with a menu shown and the line edit losing
# focus.
self.abort_when_line_edit_unfocussed = False
if self.menu:
# Hide menu temporarily to avoid re-layouting on every added item.
self.abort_when_menu_hidden = False
self.menu.hide()
self.menu.clear()
self._ensure_menu()
first_action = None
last_action = None
for choice in filtered_choices:
action = QtWidgets.QAction(choice, self.menu)
action.triggered.connect(partial(self._finish, action, choice))
action.modifiers = 0
self.menu.addAction(action)
if not first_action:
first_action = action
last_action = action
if num_omitted > 0:
action = QtWidgets.QAction("<{} not shown>".format(num_omitted),
self.menu)
action.setEnabled(False)
self.menu.addAction(action)
if self.menu_typing_filter:
self.menu.removeEventFilter(self.menu_typing_filter)
self.menu_typing_filter = _NonUpDownKeyFilter(self.menu,
self.line_edit)
self.menu.installEventFilter(self.menu_typing_filter)
if self.line_edit_up_down_filter:
self.line_edit.removeEventFilter(self.line_edit_up_down_filter)
self.line_edit_up_down_filter = _UpDownKeyFilter(
self.line_edit, self.menu, first_action, last_action)
self.line_edit.installEventFilter(self.line_edit_up_down_filter)
self.abort_when_menu_hidden = True
self.menu.show()
if first_action:
self.menu.setActiveAction(first_action)
self.menu.setFocus()
else:
self.line_edit.setFocus()
def _filter_choices(self):
"""Return a filtered and ranked list of choices based on the current
user input.
For a choice not to be filtered out, it needs to contain the entered
characters in order. Entries are further sorted by the length of the
match (i.e. preferring matches where the entered string occurrs
without interruptions), then the position of the match, and finally
lexicographically.
"""
query = self.line_edit.text()
if not query:
return [label for label, _ in self.choices]
# Find all "substring" matches of the given query in the labels,
# allowing any number of characters between each query character.
# Sort first by length of match (short matches preferred), to which the
# set weight is also applied, then by location (early in the label
# preferred), and at last alphabetically.
# TODO: More SublimeText-like heuristics taking capital letters and
# punctuation into account. Also, requiring the matches to be in order
# seems to be a bit annoying in practice.
# `re` seems to be the fastest way of doing this in CPython, even with
# all the (non-greedy) wildcards.
suggestions = []
pattern_str = ".*?".join(map(re.escape, query))
pattern = re.compile(pattern_str, flags=re.IGNORECASE)
for label, weight in self.choices:
matches = []
# Manually loop over shortest matches at each position;
# re.finditer() only returns non-overlapping matches.
pos = 0
while True:
r = pattern.search(label, pos=pos)
if not r:
break
start, stop = r.span()
matches.append((stop - start - weight, start, label))
pos = start + 1
if matches:
suggestions.append(min(matches))
return [x for _, _, x in sorted(suggestions)]
def _close(self):
if self.menu:
self.menu.close()
self.menu = None
self.update_when_text_changed = False
self.line_edit.clear()
def abort(self):
self._close()
self.aborted.emit()
def _finish(self, action, name):
self._close()
self.finished.emit(name, action.modifiers)
class _FocusEventFilter(QtCore.QObject):
"""Emits signals when focus is gained/lost."""
focus_gained = QtCore.pyqtSignal()
focus_lost = QtCore.pyqtSignal()
def eventFilter(self, obj, event):
if event.type() == QtCore.QEvent.FocusIn:
self.focus_gained.emit()
elif event.type() == QtCore.QEvent.FocusOut:
self.focus_lost.emit()
return False
class _EscapeKeyFilter(QtCore.QObject):
"""Emits a signal if the Escape key is pressed."""
escape_pressed = QtCore.pyqtSignal()
def eventFilter(self, obj, event):
if event.type() == QtCore.QEvent.KeyPress:
if event.key() == QtCore.Qt.Key_Escape:
self.escape_pressed.emit()
return False
class _UpDownKeyFilter(QtCore.QObject):
"""Handles focussing the menu when pressing up/down in the line edit."""
def __init__(self, parent, menu, first_item, last_item):
super().__init__(parent)
self.menu = menu
self.first_item = first_item
self.last_item = last_item
def eventFilter(self, obj, event):
if event.type() == QtCore.QEvent.KeyPress:
if event.key() == QtCore.Qt.Key_Down:
self.menu.setActiveAction(self.first_item)
self.menu.setFocus()
return True
if event.key() == QtCore.Qt.Key_Up:
self.menu.setActiveAction(self.last_item)
self.menu.setFocus()
return True
return False
class _NonUpDownKeyFilter(QtCore.QObject):
"""Forwards input while the menu is focussed to the line edit."""
def __init__(self, parent, target):
super().__init__(parent)
self.target = target
def eventFilter(self, obj, event):
if event.type() == QtCore.QEvent.KeyPress:
k = event.key()
if k in (QtCore.Qt.Key_Return, QtCore.Qt.Key_Enter):
action = obj.activeAction()
if action is not None:
action.modifiers = event.modifiers()
return False
if (k != QtCore.Qt.Key_Down and k != QtCore.Qt.Key_Up
and k != QtCore.Qt.Key_Enter
and k != QtCore.Qt.Key_Return):
QtWidgets.QApplication.sendEvent(self.target, event)
return True
return False