2021-04-23 15:12:29 +00:00
|
|
|
# piker: trading gear for hackers
|
|
|
|
# Copyright (C) Tyler Goodlet (in stewardship for piker0)
|
|
|
|
|
|
|
|
# This program is free software: you can redistribute it and/or modify
|
|
|
|
# it under the terms of the GNU Affero General Public License as published by
|
|
|
|
# the Free Software Foundation, either version 3 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
|
|
|
|
# This program 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 Affero General Public License for more details.
|
|
|
|
|
|
|
|
# You should have received a copy of the GNU Affero General Public License
|
|
|
|
# along with this program. If not, see <https://www.gnu.org/licenses/>.
|
|
|
|
|
|
|
|
"""
|
|
|
|
qompleterz: embeddable search and complete using trio, Qt and fuzzywuzzy.
|
|
|
|
|
|
|
|
"""
|
2021-05-15 23:35:52 +00:00
|
|
|
|
|
|
|
# link set for hackzin on this stuff:
|
|
|
|
# https://doc.qt.io/qt-5/qheaderview.html#moving-header-sections
|
|
|
|
# https://doc.qt.io/qt-5/model-view-programming.html
|
|
|
|
# https://doc.qt.io/qt-5/modelview.html
|
|
|
|
# https://doc.qt.io/qt-5/qtreeview.html#selectedIndexes
|
|
|
|
# https://doc.qt.io/qt-5/qmodelindex.html#siblingAtColumn
|
|
|
|
# https://doc.qt.io/qt-5/qitemselectionmodel.html#currentIndex
|
|
|
|
# https://www.programcreek.com/python/example/108109/PyQt5.QtWidgets.QTreeView
|
|
|
|
# https://doc.qt.io/qt-5/qsyntaxhighlighter.html
|
|
|
|
# https://github.com/qutebrowser/qutebrowser/blob/master/qutebrowser/completion/completiondelegate.py#L243
|
|
|
|
# https://forum.qt.io/topic/61343/highlight-matched-substrings-in-qstyleditemdelegate
|
|
|
|
|
2021-05-17 00:52:22 +00:00
|
|
|
from contextlib import asynccontextmanager
|
2021-05-14 11:51:42 +00:00
|
|
|
from functools import partial
|
2021-05-10 14:10:53 +00:00
|
|
|
from typing import (
|
|
|
|
List, Optional, Callable,
|
|
|
|
Awaitable, Sequence, Dict,
|
2021-05-17 00:52:22 +00:00
|
|
|
Any, AsyncIterator, Tuple,
|
2021-05-10 14:10:53 +00:00
|
|
|
)
|
2021-05-06 20:41:15 +00:00
|
|
|
# from pprint import pformat
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-16 19:40:31 +00:00
|
|
|
from fuzzywuzzy import process as fuzzy
|
|
|
|
import trio
|
2021-04-23 15:12:29 +00:00
|
|
|
from PyQt5 import QtCore, QtGui
|
|
|
|
from PyQt5 import QtWidgets
|
|
|
|
from PyQt5.QtCore import (
|
|
|
|
Qt,
|
2021-05-06 20:41:15 +00:00
|
|
|
# QSize,
|
2021-04-23 15:12:29 +00:00
|
|
|
QModelIndex,
|
|
|
|
QItemSelectionModel,
|
|
|
|
)
|
|
|
|
from PyQt5.QtGui import (
|
2021-05-15 23:35:52 +00:00
|
|
|
# QLayout,
|
2021-04-23 15:12:29 +00:00
|
|
|
QStandardItem,
|
|
|
|
QStandardItemModel,
|
|
|
|
)
|
|
|
|
from PyQt5.QtWidgets import (
|
2021-05-14 11:51:42 +00:00
|
|
|
QWidget,
|
2021-04-23 15:12:29 +00:00
|
|
|
QTreeView,
|
|
|
|
# QListWidgetItem,
|
2021-05-15 23:35:52 +00:00
|
|
|
# QAbstractScrollArea,
|
2021-04-23 15:12:29 +00:00
|
|
|
QStyledItemDelegate,
|
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
from ..log import get_logger
|
|
|
|
from ._style import (
|
|
|
|
_font,
|
|
|
|
DpiAwareFont,
|
2021-05-06 20:41:15 +00:00
|
|
|
# hcolor,
|
2021-04-23 15:12:29 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
|
|
|
log = get_logger(__name__)
|
|
|
|
|
|
|
|
|
|
|
|
class SimpleDelegate(QStyledItemDelegate):
|
|
|
|
"""
|
|
|
|
Super simple view delegate to render text in the same
|
|
|
|
font size as the search widget.
|
|
|
|
|
|
|
|
"""
|
|
|
|
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
parent=None,
|
|
|
|
font: DpiAwareFont = _font,
|
|
|
|
) -> None:
|
|
|
|
super().__init__(parent)
|
|
|
|
self.dpi_font = font
|
|
|
|
|
|
|
|
# def sizeHint(self, *args) -> QtCore.QSize:
|
|
|
|
# """
|
|
|
|
# Scale edit box to size of dpi aware font.
|
|
|
|
|
|
|
|
# """
|
|
|
|
# psh = super().sizeHint(*args)
|
|
|
|
# # psh.setHeight(self.dpi_font.px_size + 2)
|
|
|
|
|
|
|
|
# psh.setHeight(18)
|
|
|
|
# # psh.setHeight(18)
|
|
|
|
# return psh
|
|
|
|
|
|
|
|
|
|
|
|
class CompleterView(QTreeView):
|
|
|
|
|
2021-05-15 23:35:52 +00:00
|
|
|
# XXX: relevant docs links:
|
|
|
|
# - simple widget version of this:
|
|
|
|
# https://doc.qt.io/qt-5/qtreewidget.html#details
|
|
|
|
# - MVC high level instructional:
|
|
|
|
# https://doc.qt.io/qt-5/model-view-programming.html
|
|
|
|
# - MV tut:
|
|
|
|
# https://doc.qt.io/qt-5/modelview.html
|
|
|
|
|
2021-04-23 15:12:29 +00:00
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
parent=None,
|
2021-05-06 20:41:15 +00:00
|
|
|
labels: List[str] = [],
|
2021-04-23 15:12:29 +00:00
|
|
|
) -> None:
|
|
|
|
|
|
|
|
super().__init__(parent)
|
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
model = QStandardItemModel(self)
|
|
|
|
self.labels = labels
|
|
|
|
|
|
|
|
# a std "tabular" config
|
|
|
|
self.setItemDelegate(SimpleDelegate())
|
|
|
|
self.setModel(model)
|
|
|
|
self.setAlternatingRowColors(True)
|
2021-05-15 23:35:52 +00:00
|
|
|
# TODO: size this based on DPI font
|
2021-05-16 19:40:31 +00:00
|
|
|
self.setIndentation(20)
|
2021-05-06 20:41:15 +00:00
|
|
|
|
|
|
|
# self.setUniformRowHeights(True)
|
|
|
|
# self.setColumnWidth(0, 3)
|
|
|
|
|
|
|
|
# ux settings
|
|
|
|
self.setItemsExpandable(True)
|
|
|
|
self.setExpandsOnDoubleClick(False)
|
|
|
|
self.setAnimated(False)
|
|
|
|
self.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
|
|
|
2021-05-15 23:35:52 +00:00
|
|
|
# self.setVerticalBarPolicy(Qt.ScrollBarAlwaysOff)
|
|
|
|
# self.setSizeAdjustPolicy(QAbstractScrollArea.AdjustIgnored)
|
2021-05-06 20:41:15 +00:00
|
|
|
|
|
|
|
# column headers
|
|
|
|
model.setHorizontalHeaderLabels(labels)
|
|
|
|
|
2021-04-23 15:12:29 +00:00
|
|
|
self._font_size: int = 0 # pixels
|
2021-05-05 14:10:02 +00:00
|
|
|
# self._cache: Dict[str, List[str]] = {}
|
2021-04-23 15:12:29 +00:00
|
|
|
|
|
|
|
# def viewportSizeHint(self) -> QtCore.QSize:
|
|
|
|
# vps = super().viewportSizeHint()
|
|
|
|
# return QSize(vps.width(), _font.px_size * 6 * 2)
|
|
|
|
|
|
|
|
# def sizeHint(self) -> QtCore.QSize:
|
|
|
|
# """Scale completion results up to 6/16 of window.
|
|
|
|
# """
|
|
|
|
# # height = self.window().height() * 1/6
|
|
|
|
# # psh.setHeight(self.dpi_font.px_size * 6)
|
|
|
|
# # print(_font.px_size)
|
|
|
|
# height = _font.px_size * 6 * 2
|
|
|
|
# # the default here is just the vp size without scroll bar
|
|
|
|
# # https://doc.qt.io/qt-5/qabstractscrollarea.html#viewportSizeHint
|
|
|
|
# vps = self.viewportSizeHint()
|
|
|
|
# # print(f'h: {height}\n{vps}')
|
|
|
|
# # psh.setHeight(12)
|
|
|
|
# return QSize(-1, height)
|
|
|
|
|
|
|
|
def set_font_size(self, size: int = 18):
|
|
|
|
# dpi_px_size = _font.px_size
|
|
|
|
print(size)
|
|
|
|
if size < 0:
|
|
|
|
size = 16
|
|
|
|
|
|
|
|
self._font_size = size
|
|
|
|
|
|
|
|
self.setStyleSheet(f"font: {size}px")
|
|
|
|
|
|
|
|
def set_results(
|
|
|
|
self,
|
2021-05-10 14:10:53 +00:00
|
|
|
results: Dict[str, Sequence[str]],
|
2021-04-23 15:12:29 +00:00
|
|
|
) -> None:
|
|
|
|
|
|
|
|
model = self.model()
|
2021-05-06 20:41:15 +00:00
|
|
|
model.clear()
|
|
|
|
model.setHorizontalHeaderLabels(self.labels)
|
|
|
|
|
|
|
|
# TODO: wtf.. this model shit
|
|
|
|
# row_count = model.rowCount()
|
|
|
|
# if row_count > 0:
|
|
|
|
# model.removeRows(
|
|
|
|
# 0,
|
|
|
|
# row_count,
|
|
|
|
|
|
|
|
# # root index
|
|
|
|
# model.index(0, 0, QModelIndex()),
|
|
|
|
# )
|
2021-05-15 23:35:52 +00:00
|
|
|
root = model.invisibleRootItem()
|
|
|
|
|
2021-05-10 14:10:53 +00:00
|
|
|
for key, values in results.items():
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-15 23:35:52 +00:00
|
|
|
src = QStandardItem(key)
|
|
|
|
root.appendRow(src)
|
|
|
|
# self.expand(model.index(1, 0, QModelIndex()))
|
|
|
|
|
2021-05-10 14:10:53 +00:00
|
|
|
# values just needs to be sequence-like
|
|
|
|
for i, s in enumerate(values):
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-15 23:35:52 +00:00
|
|
|
# blank = QStandardItem('')
|
2021-05-10 14:10:53 +00:00
|
|
|
ix = QStandardItem(str(i))
|
|
|
|
item = QStandardItem(s)
|
|
|
|
# item.setCheckable(False)
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-10 14:10:53 +00:00
|
|
|
# Add the item to the model
|
2021-05-15 23:35:52 +00:00
|
|
|
src.appendRow([ix, item])
|
|
|
|
|
|
|
|
self.expandAll()
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-17 00:52:22 +00:00
|
|
|
# XXX: these 2 lines MUST be in sequence in order
|
|
|
|
# to get the view to show right after typing input.
|
|
|
|
sel = self.selectionModel()
|
|
|
|
sel.setCurrentIndex(
|
|
|
|
model.index(0, 0, QModelIndex()),
|
|
|
|
QItemSelectionModel.ClearAndSelect |
|
|
|
|
QItemSelectionModel.Rows
|
|
|
|
)
|
|
|
|
self.select_from_idx(model.index(0, 0, QModelIndex()))
|
|
|
|
|
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
def show_matches(self) -> None:
|
|
|
|
# print(f"SHOWING {self}")
|
|
|
|
self.show()
|
|
|
|
self.resize()
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
def resize(self):
|
|
|
|
model = self.model()
|
|
|
|
cols = model.columnCount()
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
for i in range(cols):
|
|
|
|
self.resizeColumnToContents(i)
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
# inclusive of search bar and header "rows" in pixel terms
|
2021-05-15 23:35:52 +00:00
|
|
|
rows = 100
|
|
|
|
# print(f'row count: {rows}')
|
2021-05-06 20:41:15 +00:00
|
|
|
# max_rows = 8 # 6 + search and headers
|
|
|
|
row_px = self.rowHeight(self.currentIndex())
|
|
|
|
# print(f'font_h: {font_h}\n px_height: {px_height}')
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
# TODO: probably make this more general / less hacky
|
|
|
|
self.setMinimumSize(self.width(), rows * row_px)
|
|
|
|
self.setMaximumSize(self.width(), rows * row_px)
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-15 23:35:52 +00:00
|
|
|
def select_previous(self) -> QModelIndex:
|
|
|
|
cidx = self.currentIndex()
|
|
|
|
nidx = self.indexAbove(cidx)
|
|
|
|
if nidx.parent() is QModelIndex():
|
|
|
|
nidx = self.indexAbove(cidx)
|
|
|
|
breakpoint()
|
|
|
|
|
|
|
|
return nidx
|
|
|
|
|
|
|
|
def select_next(self) -> QModelIndex:
|
|
|
|
cidx = self.currentIndex()
|
|
|
|
nidx = self.indexBelow(cidx)
|
|
|
|
if nidx.parent() is QModelIndex():
|
|
|
|
nidx = self.indexBelow(cidx)
|
|
|
|
breakpoint()
|
|
|
|
|
|
|
|
return nidx
|
|
|
|
|
|
|
|
def select_from_idx(
|
2021-05-17 00:52:22 +00:00
|
|
|
|
2021-05-15 23:35:52 +00:00
|
|
|
self,
|
|
|
|
idx: QModelIndex,
|
2021-05-17 00:52:22 +00:00
|
|
|
|
|
|
|
) -> Tuple[QModelIndex, QStandardItem]:
|
|
|
|
|
2021-05-15 23:35:52 +00:00
|
|
|
sel = self.selectionModel()
|
|
|
|
model = self.model()
|
|
|
|
|
|
|
|
# select first indented entry
|
|
|
|
if idx == model.index(0, 0):
|
|
|
|
idx = self.select_next()
|
|
|
|
|
|
|
|
sel.setCurrentIndex(
|
|
|
|
idx,
|
|
|
|
QItemSelectionModel.ClearAndSelect |
|
|
|
|
QItemSelectionModel.Rows
|
|
|
|
)
|
|
|
|
|
2021-05-17 00:52:22 +00:00
|
|
|
return idx, model.itemFromIndex(idx)
|
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
# def find_matches(
|
|
|
|
# self,
|
|
|
|
# field: str,
|
|
|
|
# txt: str,
|
|
|
|
# ) -> List[QStandardItem]:
|
|
|
|
# model = self.model()
|
|
|
|
# items = model.findItems(
|
|
|
|
# txt,
|
|
|
|
# Qt.MatchContains,
|
|
|
|
# self.field_to_col(field),
|
|
|
|
# )
|
2021-04-23 15:12:29 +00:00
|
|
|
|
|
|
|
|
2021-05-14 11:51:42 +00:00
|
|
|
class SearchBar(QtWidgets.QLineEdit):
|
2021-04-23 15:12:29 +00:00
|
|
|
|
|
|
|
def __init__(
|
2021-05-14 11:51:42 +00:00
|
|
|
|
2021-04-23 15:12:29 +00:00
|
|
|
self,
|
2021-05-14 11:51:42 +00:00
|
|
|
parent: QWidget,
|
|
|
|
parent_chart: QWidget, # noqa
|
2021-04-23 15:12:29 +00:00
|
|
|
view: Optional[CompleterView] = None,
|
|
|
|
font: DpiAwareFont = _font,
|
|
|
|
|
2021-05-14 11:51:42 +00:00
|
|
|
) -> None:
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-14 11:51:42 +00:00
|
|
|
super().__init__(parent)
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-14 11:51:42 +00:00
|
|
|
self.view: CompleterView = view
|
2021-04-23 15:12:29 +00:00
|
|
|
self.dpi_font = font
|
|
|
|
self.chart_app = parent_chart
|
|
|
|
|
|
|
|
# size it as we specify
|
2021-05-15 23:35:52 +00:00
|
|
|
# https://doc.qt.io/qt-5/qsizepolicy.html#Policy-enum
|
2021-04-23 15:12:29 +00:00
|
|
|
self.setSizePolicy(
|
|
|
|
QtWidgets.QSizePolicy.Fixed,
|
|
|
|
QtWidgets.QSizePolicy.Fixed,
|
|
|
|
)
|
|
|
|
self.setFont(font.font)
|
|
|
|
|
|
|
|
# witty bit of margin
|
|
|
|
self.setTextMargins(2, 2, 2, 2)
|
|
|
|
|
|
|
|
# self.setContextMenuPolicy(Qt.CustomContextMenu)
|
|
|
|
# self.customContextMenuRequested.connect(self.show_menu)
|
|
|
|
# self.setStyleSheet(f"font: 18px")
|
|
|
|
|
|
|
|
def focus(self) -> None:
|
|
|
|
self.clear()
|
|
|
|
self.show()
|
|
|
|
self.setFocus()
|
|
|
|
|
|
|
|
def show(self) -> None:
|
|
|
|
super().show()
|
2021-05-06 20:41:15 +00:00
|
|
|
self.view.show_matches()
|
2021-04-23 15:12:29 +00:00
|
|
|
|
|
|
|
def sizeHint(self) -> QtCore.QSize:
|
|
|
|
"""
|
|
|
|
Scale edit box to size of dpi aware font.
|
|
|
|
|
|
|
|
"""
|
|
|
|
psh = super().sizeHint()
|
|
|
|
psh.setHeight(self.dpi_font.px_size + 2)
|
2021-05-14 11:51:42 +00:00
|
|
|
psh.setWidth(6*6*6)
|
2021-04-23 15:12:29 +00:00
|
|
|
return psh
|
|
|
|
|
|
|
|
def unfocus(self) -> None:
|
|
|
|
self.hide()
|
|
|
|
self.clearFocus()
|
|
|
|
|
|
|
|
if self.view:
|
|
|
|
self.view.hide()
|
|
|
|
|
2021-05-05 14:10:02 +00:00
|
|
|
|
2021-05-11 21:26:27 +00:00
|
|
|
_search_active: trio.Event = trio.Event()
|
2021-05-10 14:10:53 +00:00
|
|
|
_search_enabled: bool = False
|
2021-05-05 14:10:02 +00:00
|
|
|
|
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
async def fill_results(
|
2021-05-11 21:26:27 +00:00
|
|
|
|
2021-05-14 11:51:42 +00:00
|
|
|
search: SearchBar,
|
2021-05-06 20:41:15 +00:00
|
|
|
symsearch: Callable[..., Awaitable],
|
2021-05-05 14:10:02 +00:00
|
|
|
recv_chan: trio.abc.ReceiveChannel,
|
2021-05-17 00:52:22 +00:00
|
|
|
# cached_symbols: Dict[str,
|
2021-05-14 11:51:42 +00:00
|
|
|
pause_time: float = 0.0616,
|
2021-05-11 21:26:27 +00:00
|
|
|
|
2021-05-05 14:10:02 +00:00
|
|
|
) -> None:
|
2021-05-10 14:10:53 +00:00
|
|
|
"""Task to search through providers and fill in possible
|
|
|
|
completion results.
|
2021-05-05 14:10:02 +00:00
|
|
|
|
2021-05-10 14:10:53 +00:00
|
|
|
"""
|
2021-05-11 21:26:27 +00:00
|
|
|
global _search_active, _search_enabled
|
2021-05-10 14:10:53 +00:00
|
|
|
|
2021-05-14 11:51:42 +00:00
|
|
|
bar = search.bar
|
|
|
|
view = bar.view
|
|
|
|
sel = bar.view.selectionModel()
|
|
|
|
model = bar.view.model()
|
2021-05-05 14:10:02 +00:00
|
|
|
|
2021-05-11 21:26:27 +00:00
|
|
|
last_search_text = ''
|
2021-05-14 11:51:42 +00:00
|
|
|
last_text = bar.text()
|
2021-05-11 21:26:27 +00:00
|
|
|
repeats = 0
|
|
|
|
|
|
|
|
while True:
|
|
|
|
|
2021-05-14 11:51:42 +00:00
|
|
|
last_text = bar.text()
|
2021-05-11 21:26:27 +00:00
|
|
|
await _search_active.wait()
|
|
|
|
|
|
|
|
with trio.move_on_after(pause_time) as cs:
|
|
|
|
# cs.shield = True
|
|
|
|
pattern = await recv_chan.receive()
|
|
|
|
print(pattern)
|
|
|
|
|
|
|
|
# during fast multiple key inputs, wait until a pause
|
|
|
|
# (in typing) to initiate search
|
|
|
|
if not cs.cancelled_caught:
|
|
|
|
log.debug(f'Ignoring fast input for {pattern}')
|
|
|
|
continue
|
|
|
|
|
2021-05-14 11:51:42 +00:00
|
|
|
text = bar.text()
|
2021-05-11 21:26:27 +00:00
|
|
|
print(f'search: {text}')
|
|
|
|
|
|
|
|
if not text:
|
|
|
|
print('idling')
|
|
|
|
_search_active = trio.Event()
|
|
|
|
continue
|
|
|
|
|
|
|
|
if text == last_text:
|
|
|
|
repeats += 1
|
|
|
|
|
|
|
|
if repeats > 1:
|
|
|
|
_search_active = trio.Event()
|
|
|
|
repeats = 0
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-10 14:10:53 +00:00
|
|
|
if not _search_enabled:
|
2021-05-11 21:26:27 +00:00
|
|
|
print('search not ENABLED?')
|
2021-05-10 14:10:53 +00:00
|
|
|
continue
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-11 21:26:27 +00:00
|
|
|
if last_search_text and last_search_text == text:
|
|
|
|
continue
|
|
|
|
|
|
|
|
log.debug(f'Search req for {text}')
|
|
|
|
|
|
|
|
last_search_text = text
|
|
|
|
results = await symsearch(text)
|
|
|
|
log.debug(f'Received search result {results}')
|
2021-05-10 14:10:53 +00:00
|
|
|
|
2021-05-11 21:26:27 +00:00
|
|
|
if results and _search_enabled:
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
# TODO: indented branch results for each provider
|
2021-05-10 14:10:53 +00:00
|
|
|
view.set_results(results)
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-10 14:10:53 +00:00
|
|
|
# XXX: these 2 lines MUST be in sequence in order
|
|
|
|
# to get the view to show right after typing input.
|
2021-05-17 00:52:22 +00:00
|
|
|
# ensure we select first indented entry
|
|
|
|
# view.select_from_idx(model.index(0, 0, QModelIndex()))
|
|
|
|
|
|
|
|
# sel.setCurrentIndex(
|
|
|
|
# model.index(0, 0, QModelIndex()),
|
|
|
|
# QItemSelectionModel.ClearAndSelect |
|
|
|
|
# QItemSelectionModel.Rows
|
|
|
|
# )
|
|
|
|
|
2021-05-14 11:51:42 +00:00
|
|
|
bar.show()
|
|
|
|
|
2021-05-17 00:52:22 +00:00
|
|
|
# # ensure we select first indented entry
|
|
|
|
# view.select_from_idx(sel.currentIndex())
|
2021-05-15 23:35:52 +00:00
|
|
|
|
2021-05-14 11:51:42 +00:00
|
|
|
|
|
|
|
class SearchWidget(QtGui.QWidget):
|
|
|
|
def __init__(
|
|
|
|
self,
|
|
|
|
chart_space: 'ChartSpace', # type: ignore # noqa
|
2021-05-15 23:35:52 +00:00
|
|
|
columns: List[str] = ['src', 'symbol'],
|
2021-05-14 11:51:42 +00:00
|
|
|
parent=None,
|
|
|
|
):
|
|
|
|
super().__init__(parent or chart_space)
|
|
|
|
|
|
|
|
# size it as we specify
|
|
|
|
self.setSizePolicy(
|
|
|
|
QtWidgets.QSizePolicy.Fixed,
|
2021-05-15 23:35:52 +00:00
|
|
|
QtWidgets.QSizePolicy.Expanding,
|
2021-05-14 11:51:42 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
self.chart_app = chart_space
|
|
|
|
self.vbox = QtGui.QVBoxLayout(self)
|
|
|
|
self.vbox.setContentsMargins(0, 0, 0, 0)
|
2021-05-15 23:35:52 +00:00
|
|
|
self.vbox.setSpacing(4)
|
|
|
|
|
|
|
|
# https://doc.qt.io/qt-5/qlayout.html#SizeConstraint-enum
|
|
|
|
# self.vbox.setSizeConstraint(QLayout.SetMaximumSize)
|
2021-05-14 11:51:42 +00:00
|
|
|
|
|
|
|
self.view = CompleterView(
|
|
|
|
parent=self,
|
|
|
|
labels=columns,
|
|
|
|
)
|
|
|
|
self.bar = SearchBar(
|
|
|
|
parent=self,
|
|
|
|
parent_chart=chart_space,
|
|
|
|
view=self.view,
|
|
|
|
)
|
|
|
|
self.vbox.addWidget(self.bar)
|
|
|
|
self.vbox.setAlignment(self.bar, Qt.AlignTop | Qt.AlignLeft)
|
|
|
|
self.vbox.addWidget(self.bar.view)
|
|
|
|
self.vbox.setAlignment(self.view, Qt.AlignTop | Qt.AlignLeft)
|
2021-05-05 14:10:02 +00:00
|
|
|
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-17 00:52:22 +00:00
|
|
|
def focus(self) -> None:
|
|
|
|
# fill cache list
|
|
|
|
self.view.set_results({'cache': list(self.chart_app._chart_cache)})
|
|
|
|
self.bar.focus()
|
|
|
|
|
|
|
|
|
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
async def handle_keyboard_input(
|
2021-05-10 14:10:53 +00:00
|
|
|
|
2021-05-16 19:40:31 +00:00
|
|
|
# chart: 'ChartSpace', # type: igore # noqa
|
2021-05-14 11:51:42 +00:00
|
|
|
search: SearchWidget,
|
2021-05-06 20:41:15 +00:00
|
|
|
recv_chan: trio.abc.ReceiveChannel,
|
2021-05-14 11:51:42 +00:00
|
|
|
keyboard_pause_period: float = 0.0616,
|
2021-05-10 14:10:53 +00:00
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
) -> None:
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-11 21:26:27 +00:00
|
|
|
global _search_active, _search_enabled
|
2021-05-10 14:10:53 +00:00
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
# startup
|
2021-05-14 11:51:42 +00:00
|
|
|
bar = search.bar
|
|
|
|
view = bar.view
|
|
|
|
view.set_font_size(bar.dpi_font.px_size)
|
2021-05-06 20:41:15 +00:00
|
|
|
model = view.model()
|
|
|
|
nidx = cidx = view.currentIndex()
|
|
|
|
sel = view.selectionModel()
|
|
|
|
|
2021-05-17 00:52:22 +00:00
|
|
|
symsearch = get_multi_search()
|
2021-05-06 20:41:15 +00:00
|
|
|
send, recv = trio.open_memory_channel(16)
|
|
|
|
|
|
|
|
async with trio.open_nursery() as n:
|
2021-05-10 14:10:53 +00:00
|
|
|
# TODO: async debouncing?
|
2021-05-06 20:41:15 +00:00
|
|
|
n.start_soon(
|
2021-05-14 11:51:42 +00:00
|
|
|
partial(
|
|
|
|
fill_results,
|
|
|
|
search,
|
|
|
|
symsearch,
|
|
|
|
recv,
|
|
|
|
pause_time=keyboard_pause_period,
|
|
|
|
)
|
2021-05-06 20:41:15 +00:00
|
|
|
)
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-10 14:10:53 +00:00
|
|
|
async for key, mods, txt in recv_chan:
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
log.debug(f'key: {key}, mods: {mods}, txt: {txt}')
|
2021-05-15 23:35:52 +00:00
|
|
|
# parent = view.currentIndex()
|
|
|
|
cidx = sel.currentIndex()
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
ctrl = False
|
|
|
|
if mods == Qt.ControlModifier:
|
|
|
|
ctrl = True
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
if key in (Qt.Key_Enter, Qt.Key_Return):
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-15 23:35:52 +00:00
|
|
|
# TODO: get rid of this hard coded column -> 1
|
2021-05-16 19:40:31 +00:00
|
|
|
# and use the ``CompleterView`` schema/settings
|
|
|
|
# to figure out the desired field(s)
|
2021-05-15 23:35:52 +00:00
|
|
|
# https://doc.qt.io/qt-5/qstandarditemmodel.html#itemFromIndex
|
|
|
|
node = model.itemFromIndex(cidx.siblingAtColumn(1))
|
2021-05-10 14:10:53 +00:00
|
|
|
if node:
|
|
|
|
value = node.text()
|
2021-05-15 23:35:52 +00:00
|
|
|
# print(f' value: {value}')
|
2021-05-10 14:10:53 +00:00
|
|
|
else:
|
|
|
|
continue
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
log.info(f'Requesting symbol: {value}')
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
app = search.chart_app
|
|
|
|
search.chart_app.load_symbol(
|
|
|
|
app.linkedcharts.symbol.brokers[0],
|
|
|
|
value,
|
|
|
|
'info',
|
|
|
|
)
|
|
|
|
|
2021-05-11 21:26:27 +00:00
|
|
|
_search_enabled = False
|
2021-05-06 20:41:15 +00:00
|
|
|
# release kb control of search bar
|
2021-05-14 11:51:42 +00:00
|
|
|
search.bar.unfocus()
|
2021-05-06 20:41:15 +00:00
|
|
|
continue
|
|
|
|
|
|
|
|
# selection tips:
|
2021-05-10 14:10:53 +00:00
|
|
|
# - get parent node: search.index(row, 0)
|
|
|
|
# - first node index: index = search.index(0, 0, parent)
|
|
|
|
# - root node index: index = search.index(0, 0, QModelIndex())
|
2021-05-06 20:41:15 +00:00
|
|
|
|
2021-05-10 14:10:53 +00:00
|
|
|
# we're in select mode or cancelling
|
2021-05-06 20:41:15 +00:00
|
|
|
if ctrl:
|
2021-05-10 14:10:53 +00:00
|
|
|
# cancel and close
|
2021-05-06 20:41:15 +00:00
|
|
|
if key == Qt.Key_C:
|
2021-05-14 11:51:42 +00:00
|
|
|
search.bar.unfocus()
|
2021-05-06 20:41:15 +00:00
|
|
|
|
|
|
|
# kill the search and focus back on main chart
|
|
|
|
if search.chart_app:
|
|
|
|
search.chart_app.linkedcharts.focus()
|
|
|
|
|
|
|
|
continue
|
|
|
|
|
|
|
|
# result selection nav
|
|
|
|
if key in (Qt.Key_K, Qt.Key_J):
|
2021-05-11 21:26:27 +00:00
|
|
|
_search_enabled = False
|
2021-05-06 20:41:15 +00:00
|
|
|
|
|
|
|
if key == Qt.Key_K:
|
2021-05-15 23:35:52 +00:00
|
|
|
nidx = view.select_previous()
|
2021-05-06 20:41:15 +00:00
|
|
|
|
|
|
|
elif key == Qt.Key_J:
|
2021-05-15 23:35:52 +00:00
|
|
|
nidx = view.select_next()
|
2021-05-06 20:41:15 +00:00
|
|
|
|
|
|
|
# select row without selecting.. :eye_rollzz:
|
|
|
|
# https://doc.qt.io/qt-5/qabstractitemview.html#setCurrentIndex
|
|
|
|
if nidx.isValid():
|
2021-05-17 00:52:22 +00:00
|
|
|
i, item = view.select_from_idx(nidx)
|
|
|
|
|
|
|
|
if item:
|
|
|
|
parent_item = item.parent()
|
|
|
|
if parent_item and parent_item.text() == 'cache':
|
|
|
|
node = model.itemFromIndex(i.siblingAtColumn(1))
|
|
|
|
if node:
|
|
|
|
value = node.text()
|
|
|
|
print(f'cache selection')
|
|
|
|
search.chart_app.load_symbol(
|
|
|
|
app.linkedcharts.symbol.brokers[0],
|
|
|
|
value,
|
|
|
|
'info',
|
|
|
|
)
|
|
|
|
|
2021-05-06 20:41:15 +00:00
|
|
|
else:
|
2021-05-11 21:26:27 +00:00
|
|
|
# relay to completer task
|
|
|
|
_search_enabled = True
|
2021-05-14 11:51:42 +00:00
|
|
|
send.send_nowait(search.bar.text())
|
2021-05-11 21:26:27 +00:00
|
|
|
_search_active.set()
|
2021-04-23 15:12:29 +00:00
|
|
|
|
|
|
|
|
2021-05-17 00:52:22 +00:00
|
|
|
async def search_simple_dict(
|
|
|
|
text: str,
|
|
|
|
source: dict,
|
|
|
|
) -> Dict[str, Any]:
|
|
|
|
|
|
|
|
# matches_per_src = {}
|
|
|
|
|
|
|
|
# for source, data in source.items():
|
|
|
|
|
|
|
|
# search routine can be specified as a function such
|
|
|
|
# as in the case of the current app's local symbol cache
|
|
|
|
matches = fuzzy.extractBests(
|
|
|
|
text,
|
|
|
|
source.keys(),
|
|
|
|
score_cutoff=90,
|
|
|
|
)
|
|
|
|
|
|
|
|
return [item[0] for item in matches]
|
|
|
|
|
|
|
|
|
|
|
|
# cache of provider names to async search routines
|
|
|
|
_searcher_cache: Dict[str, Callable[..., Awaitable]] = {}
|
|
|
|
|
|
|
|
|
|
|
|
def get_multi_search() -> Callable[..., Awaitable]:
|
|
|
|
|
|
|
|
global _searcher_cache
|
|
|
|
|
|
|
|
async def multisearcher(
|
|
|
|
pattern: str,
|
|
|
|
) -> dict:
|
|
|
|
|
|
|
|
matches = {}
|
|
|
|
|
|
|
|
async def pack_matches(
|
|
|
|
provider: str,
|
|
|
|
pattern: str,
|
|
|
|
search: Callable[..., Awaitable[dict]],
|
|
|
|
) -> None:
|
|
|
|
log.debug(f'Searching {provider} for "{pattern}"')
|
|
|
|
results = await search(pattern)
|
|
|
|
if results:
|
|
|
|
matches[provider] = results
|
|
|
|
|
|
|
|
# TODO: make this an async stream?
|
|
|
|
async with trio.open_nursery() as n:
|
|
|
|
|
|
|
|
for brokername, search in _searcher_cache.items():
|
|
|
|
n.start_soon(pack_matches, brokername, pattern, search)
|
|
|
|
|
|
|
|
return matches
|
|
|
|
|
|
|
|
return multisearcher
|
|
|
|
|
|
|
|
|
|
|
|
@asynccontextmanager
|
|
|
|
async def register_symbol_search(
|
|
|
|
|
|
|
|
provider_name: str,
|
|
|
|
search_routine: Callable,
|
|
|
|
|
|
|
|
) -> AsyncIterator[dict]:
|
|
|
|
|
|
|
|
global _searcher_cache
|
|
|
|
|
|
|
|
# deliver search func to consumer
|
|
|
|
try:
|
|
|
|
_searcher_cache[provider_name] = search_routine
|
|
|
|
yield search_routine
|
|
|
|
finally:
|
|
|
|
_searcher_cache.pop(provider_name)
|
|
|
|
|
|
|
|
|
|
|
|
# if __name__ == '__main__':
|
|
|
|
|
|
|
|
# TODO: simple standalone widget testing script (moreso
|
|
|
|
# for if/when we decide to expose this module as a standalone
|
|
|
|
# repo/project).
|
2021-04-23 15:12:29 +00:00
|
|
|
|
2021-05-17 00:52:22 +00:00
|
|
|
# import sys
|
2021-04-23 15:12:29 +00:00
|
|
|
# local testing of **just** the search UI
|
2021-05-17 00:52:22 +00:00
|
|
|
# app = QtWidgets.QApplication(sys.argv)
|
|
|
|
|
|
|
|
# syms = [
|
|
|
|
# 'XMRUSD',
|
|
|
|
# 'XBTUSD',
|
|
|
|
# 'ETHUSD',
|
|
|
|
# 'XMRXBT',
|
|
|
|
# 'XDGUSD',
|
|
|
|
# 'ADAUSD',
|
|
|
|
# ]
|
|
|
|
# # search.show()
|
|
|
|
|
|
|
|
# sys.exit(app.exec_())
|