| 
									
										
										
										
											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/>. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | """
 | 
					
						
							| 
									
										
										
										
											2023-09-13 15:57:02 +00:00
										 |  |  | qompleterz: embeddable search and complete using trio, Qt and rapidfuzz. | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | """
 | 
					
						
							| 
									
										
										
										
											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-26 17:49:14 +00:00
										 |  |  | from collections import defaultdict | 
					
						
							| 
									
										
										
										
											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 ( | 
					
						
							| 
									
										
										
										
											2022-09-12 13:58:11 +00:00
										 |  |  |     Callable, | 
					
						
							|  |  |  |     Awaitable, | 
					
						
							|  |  |  |     Sequence, | 
					
						
							|  |  |  |     Any, | 
					
						
							|  |  |  |     AsyncIterator, | 
					
						
							|  |  |  |     Iterator, | 
					
						
							| 
									
										
										
										
											2021-05-10 14:10:53 +00:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  | import time | 
					
						
							| 
									
										
										
										
											2023-10-01 21:46:46 +00:00
										 |  |  | from pprint import pformat | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-09-13 15:57:02 +00:00
										 |  |  | from rapidfuzz import process as fuzzy | 
					
						
							| 
									
										
										
										
											2021-05-16 19:40:31 +00:00
										 |  |  | import trio | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  | from trio_typing import TaskStatus | 
					
						
							| 
									
										
										
										
											2024-05-01 18:33:10 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | from piker.ui.qt import ( | 
					
						
							|  |  |  |     size_policy, | 
					
						
							|  |  |  |     align_flag, | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  |     Qt, | 
					
						
							| 
									
										
										
										
											2024-05-01 18:33:10 +00:00
										 |  |  |     QtCore, | 
					
						
							|  |  |  |     QtWidgets, | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  |     QModelIndex, | 
					
						
							|  |  |  |     QItemSelectionModel, | 
					
						
							| 
									
										
										
										
											2021-05-15 23:35:52 +00:00
										 |  |  |     # QLayout, | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  |     QStandardItem, | 
					
						
							|  |  |  |     QStandardItemModel, | 
					
						
							| 
									
										
										
										
											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-07-25 19:07:26 +00:00
										 |  |  |     # QStyledItemDelegate, | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | ) | 
					
						
							|  |  |  | from ..log import get_logger | 
					
						
							|  |  |  | from ._style import ( | 
					
						
							|  |  |  |     _font, | 
					
						
							| 
									
										
										
										
											2021-07-26 15:33:37 +00:00
										 |  |  |     hcolor, | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | ) | 
					
						
							| 
									
										
										
										
											2021-09-23 14:10:59 +00:00
										 |  |  | from ._forms import Edit, FontScaledDelegate | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | log = get_logger(__name__) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class CompleterView(QTreeView): | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-24 20:04:58 +00:00
										 |  |  |     mode_name: str = 'search-nav' | 
					
						
							| 
									
										
										
										
											2021-05-30 12:45:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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-05-27 15:40:08 +00:00
										 |  |  |     # - custome header view (for doing stuff like we have in  kivy?): | 
					
						
							|  |  |  |     #   https://doc.qt.io/qt-5/qheaderview.html#moving-header-sections | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # TODO: selection model stuff for eventual aggregate feeds | 
					
						
							|  |  |  |     # charting and mgmt; | 
					
						
							|  |  |  |     # https://doc.qt.io/qt-5/qabstractitemview.html#setSelectionModel | 
					
						
							|  |  |  |     # https://doc.qt.io/qt-5/qitemselectionmodel.html | 
					
						
							|  |  |  |     # https://doc.qt.io/qt-5/modelview.html#3-2-working-with-selections | 
					
						
							|  |  |  |     # https://doc.qt.io/qt-5/model-view-programming.html#handling-selections-of-items | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     # TODO: mouse extended handling: | 
					
						
							|  |  |  |     # https://doc.qt.io/qt-5/qabstractitemview.html#entered | 
					
						
							| 
									
										
										
										
											2021-05-15 23:35:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  |     def __init__( | 
					
						
							|  |  |  |         self, | 
					
						
							|  |  |  |         parent=None, | 
					
						
							| 
									
										
										
										
											2021-08-03 13:47:29 +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 | 
					
						
							| 
									
										
										
										
											2021-07-25 19:07:26 +00:00
										 |  |  |         self.setItemDelegate(FontScaledDelegate(self)) | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  |         self.setModel(model) | 
					
						
							|  |  |  |         self.setAlternatingRowColors(True) | 
					
						
							| 
									
										
										
										
											2021-05-15 23:35:52 +00:00
										 |  |  |         # TODO: size this based on DPI font | 
					
						
							| 
									
										
										
										
											2021-09-12 23:36:50 +00:00
										 |  |  |         self.setIndentation(_font.px_size) | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-12 13:58:11 +00:00
										 |  |  |         self.setUniformRowHeights(True) | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  |         # self.setColumnWidth(0, 3) | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  |         # self.setVerticalBarPolicy(Qt.ScrollBarAlwaysOff) | 
					
						
							|  |  |  |         # self.setSizeAdjustPolicy(QAbstractScrollArea.AdjustIgnored) | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # ux settings | 
					
						
							| 
									
										
										
										
											2021-09-19 22:37:46 +00:00
										 |  |  |         self.setSizePolicy( | 
					
						
							| 
									
										
										
										
											2024-05-01 18:33:10 +00:00
										 |  |  |             size_policy.Expanding, | 
					
						
							|  |  |  |             size_policy.Expanding, | 
					
						
							| 
									
										
										
										
											2021-09-19 22:37:46 +00:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  |         self.setItemsExpandable(True) | 
					
						
							|  |  |  |         self.setExpandsOnDoubleClick(False) | 
					
						
							|  |  |  |         self.setAnimated(False) | 
					
						
							|  |  |  |         self.setHorizontalScrollBarPolicy(Qt.ScrollBarAlwaysOff) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # column headers | 
					
						
							|  |  |  |         model.setHorizontalHeaderLabels(labels) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  |         self._font_size: int = 0  # pixels | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |         self._init: bool = False | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |     async def on_pressed( | 
					
						
							|  |  |  |         self, | 
					
						
							|  |  |  |         idx: QModelIndex, | 
					
						
							|  |  |  |     ) -> None: | 
					
						
							| 
									
										
										
										
											2022-09-07 21:50:10 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Mouse pressed on view handler. | 
					
						
							| 
									
										
										
										
											2021-05-27 03:47:20 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-30 12:45:40 +00:00
										 |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2021-05-27 03:47:20 +00:00
										 |  |  |         search = self.parent() | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         await search.chart_current_item( | 
					
						
							|  |  |  |             clear_to_cache=True, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # XXX: this causes Qt to hang and segfault..lovely | 
					
						
							|  |  |  |         # self.show_cache_entries( | 
					
						
							|  |  |  |         #     only=True, | 
					
						
							|  |  |  |         #     keep_current_item_selected=True, | 
					
						
							|  |  |  |         # ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-27 03:47:20 +00:00
										 |  |  |         search.focus() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  |     def set_font_size(self, size: int = 18): | 
					
						
							| 
									
										
										
										
											2021-05-27 03:47:20 +00:00
										 |  |  |         # print(size) | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  |         if size < 0: | 
					
						
							|  |  |  |             size = 16 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self._font_size = size | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.setStyleSheet(f"font: {size}px") | 
					
						
							| 
									
										
										
										
											2022-02-10 19:22:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |     def resize_to_results( | 
					
						
							|  |  |  |         self, | 
					
						
							| 
									
										
										
										
											2023-02-21 14:14:26 +00:00
										 |  |  |         w: float | None = 0, | 
					
						
							|  |  |  |         h: float | None = None, | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |     ) -> None: | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  |         model = self.model() | 
					
						
							|  |  |  |         cols = model.columnCount() | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |         cidx = self.selectionModel().currentIndex() | 
					
						
							| 
									
										
										
										
											2022-09-12 13:58:11 +00:00
										 |  |  |         rows = model.rowCount() | 
					
						
							|  |  |  |         self.expandAll() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-12 17:45:48 +00:00
										 |  |  |         # compute the approx height in pixels needed to include | 
					
						
							|  |  |  |         # all result rows in view. | 
					
						
							| 
									
										
										
										
											2022-09-12 19:37:16 +00:00
										 |  |  |         row_h = rows_h = self.rowHeight(cidx) * (rows + 1) | 
					
						
							| 
									
										
										
										
											2022-09-12 13:58:11 +00:00
										 |  |  |         for idx, item in self.iter_df_rows(): | 
					
						
							|  |  |  |             row_h = self.rowHeight(idx) | 
					
						
							|  |  |  |             rows_h += row_h | 
					
						
							|  |  |  |             # print(f'row_h: {row_h}\nrows_h: {rows_h}') | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-12 17:45:48 +00:00
										 |  |  |             # TODO: could we just break early here on detection | 
					
						
							|  |  |  |             # of ``rows_h >= h``? | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-02-10 19:22:46 +00:00
										 |  |  |         col_w_tot = 0 | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  |         for i in range(cols): | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |             # only slap in a rows's height's worth | 
					
						
							|  |  |  |             # of padding once at startup.. no idea | 
					
						
							|  |  |  |             if ( | 
					
						
							|  |  |  |                 not self._init | 
					
						
							|  |  |  |                 and row_h | 
					
						
							|  |  |  |             ): | 
					
						
							|  |  |  |                 col_w_tot = row_h | 
					
						
							|  |  |  |                 self._init = True | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  |             self.resizeColumnToContents(i) | 
					
						
							| 
									
										
										
										
											2022-02-10 19:22:46 +00:00
										 |  |  |             col_w_tot += self.columnWidth(i) | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-12 17:45:48 +00:00
										 |  |  |         # NOTE: if the heigh `h` set here is **too large** then the | 
					
						
							|  |  |  |         # resize event will perpetually trigger as the window causes | 
					
						
							|  |  |  |         # some kind of recompute of callbacks.. so we have to ensure | 
					
						
							|  |  |  |         # it's limited. | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |         if h: | 
					
						
							|  |  |  |             h: int = round(h) | 
					
						
							| 
									
										
										
										
											2022-09-12 13:58:11 +00:00
										 |  |  |             abs_mx = round(0.91 * h) | 
					
						
							|  |  |  |             self.setMaximumHeight(abs_mx) | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-12 13:58:11 +00:00
										 |  |  |             if rows_h <= abs_mx: | 
					
						
							|  |  |  |                 # self.setMinimumHeight(rows_h) | 
					
						
							|  |  |  |                 self.setMinimumHeight(rows_h) | 
					
						
							|  |  |  |                 # self.setFixedHeight(rows_h) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 self.setMinimumHeight(abs_mx) | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # dyncamically size to width of longest result seen | 
					
						
							|  |  |  |         curr_w = self.width() | 
					
						
							|  |  |  |         if curr_w < col_w_tot: | 
					
						
							|  |  |  |             self.setMinimumWidth(col_w_tot) | 
					
						
							| 
									
										
										
										
											2022-02-10 19:22:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-19 22:37:46 +00:00
										 |  |  |         self.update() | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |     def is_selecting_d1(self) -> bool: | 
					
						
							|  |  |  |         cidx = self.selectionModel().currentIndex() | 
					
						
							|  |  |  |         return cidx.parent() == QModelIndex() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |     def previous_index(self) -> QModelIndex: | 
					
						
							| 
									
										
										
										
											2021-05-15 23:35:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |         cidx = self.selectionModel().currentIndex() | 
					
						
							|  |  |  |         one_above = self.indexAbove(cidx) | 
					
						
							| 
									
										
										
										
											2021-05-15 23:35:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |         if one_above.parent() == QModelIndex(): | 
					
						
							|  |  |  |             # if the next node up's parent is the root we don't want to select | 
					
						
							|  |  |  |             # the next node up since it's a top level node and we only | 
					
						
							|  |  |  |             # select entries depth >= 2. | 
					
						
							| 
									
										
										
										
											2021-05-15 23:35:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |             # see if one more up is not the root and we can select it. | 
					
						
							|  |  |  |             two_above = self.indexAbove(one_above) | 
					
						
							|  |  |  |             if two_above != QModelIndex(): | 
					
						
							|  |  |  |                 return two_above | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 return cidx | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return one_above  # just next up | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def next_index(self) -> QModelIndex: | 
					
						
							|  |  |  |         cidx = self.selectionModel().currentIndex() | 
					
						
							|  |  |  |         one_below = self.indexBelow(cidx) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if one_below.parent() == QModelIndex(): | 
					
						
							|  |  |  |             # if the next node up's parent is the root we don't want to select | 
					
						
							|  |  |  |             # the next node up since it's a top level node and we only | 
					
						
							|  |  |  |             # select entries depth >= 2. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             # see if one more up is not the root and we can select it. | 
					
						
							|  |  |  |             two_below = self.indexBelow(one_below) | 
					
						
							|  |  |  |             if two_below != QModelIndex(): | 
					
						
							|  |  |  |                 return two_below | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 return cidx | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 18:28:08 +00:00
										 |  |  |         return one_below  # just next down | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  |     def select_from_idx( | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self, | 
					
						
							|  |  |  |         idx: QModelIndex, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ) -> QStandardItem: | 
					
						
							| 
									
										
										
										
											2022-02-11 13:32:28 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Select and return the item at index ``idx``. | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         sel = self.selectionModel() | 
					
						
							|  |  |  |         model = self.model() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         sel.setCurrentIndex( | 
					
						
							|  |  |  |             idx, | 
					
						
							|  |  |  |             QItemSelectionModel.ClearAndSelect | | 
					
						
							|  |  |  |             QItemSelectionModel.Rows | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return model.itemFromIndex(idx) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def select_first(self) -> QStandardItem: | 
					
						
							| 
									
										
										
										
											2022-02-11 13:32:28 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Select the first depth >= 2 entry from the completer tree and | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |         return its item. | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         # ensure we're **not** selecting the first level parent node and | 
					
						
							|  |  |  |         # instead its child. | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |         model = self.model() | 
					
						
							|  |  |  |         for idx, item in self.iter_d1(): | 
					
						
							|  |  |  |             if model.rowCount(idx) == 0: | 
					
						
							|  |  |  |                 continue | 
					
						
							|  |  |  |             else: | 
					
						
							|  |  |  |                 return self.select_from_idx(self.indexBelow(idx)) | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def select_next(self) -> QStandardItem: | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |         idx = self.next_index() | 
					
						
							|  |  |  |         assert idx.isValid() | 
					
						
							|  |  |  |         return self.select_from_idx(idx) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  |     def select_previous(self) -> QStandardItem: | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |         idx = self.previous_index() | 
					
						
							|  |  |  |         assert idx.isValid() | 
					
						
							|  |  |  |         return self.select_from_idx(idx) | 
					
						
							| 
									
										
										
										
											2021-05-15 23:35:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 18:28:08 +00:00
										 |  |  |     def next_section(self, direction: str = 'down') -> QModelIndex: | 
					
						
							|  |  |  |         cidx = start_idx = self.selectionModel().currentIndex() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # step up levels to depth == 1 | 
					
						
							|  |  |  |         while cidx.parent() != QModelIndex(): | 
					
						
							|  |  |  |             cidx = cidx.parent() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # move to next section in `direction` | 
					
						
							|  |  |  |         op = {'up': -1, 'down': +1}[direction] | 
					
						
							|  |  |  |         next_row = cidx.row() + op | 
					
						
							|  |  |  |         nidx = self.model().index(next_row, cidx.column(), QModelIndex()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # do nothing, if there is no valid "next" section | 
					
						
							|  |  |  |         if not nidx.isValid(): | 
					
						
							|  |  |  |             return self.select_from_idx(start_idx) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # go to next selectable child item | 
					
						
							|  |  |  |         self.select_from_idx(nidx) | 
					
						
							|  |  |  |         return self.select_next() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |     def iter_d1( | 
					
						
							| 
									
										
										
										
											2021-05-15 23:35:52 +00:00
										 |  |  |         self, | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |     ) -> tuple[QModelIndex, QStandardItem]: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         model = self.model() | 
					
						
							|  |  |  |         isections = model.rowCount() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # much thanks to following code to figure out breadth-first | 
					
						
							|  |  |  |         # traversing from the root node: | 
					
						
							|  |  |  |         # https://stackoverflow.com/a/33126689 | 
					
						
							|  |  |  |         for i in range(isections): | 
					
						
							|  |  |  |             idx = model.index(i, 0, QModelIndex()) | 
					
						
							|  |  |  |             item = model.itemFromIndex(idx) | 
					
						
							|  |  |  |             yield idx, item | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-12 13:58:11 +00:00
										 |  |  |     def iter_df_rows( | 
					
						
							|  |  |  |         self, | 
					
						
							|  |  |  |         iparent: QModelIndex = QModelIndex(), | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ) -> Iterator[tuple[QModelIndex, QStandardItem]]: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         model = self.model() | 
					
						
							|  |  |  |         isections = model.rowCount(iparent) | 
					
						
							|  |  |  |         for i in range(isections): | 
					
						
							|  |  |  |             idx = model.index(i, 0, iparent) | 
					
						
							|  |  |  |             item = model.itemFromIndex(idx) | 
					
						
							|  |  |  |             yield idx, item | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if model.hasChildren(idx): | 
					
						
							|  |  |  |                 # recursively yield child items depth-first | 
					
						
							|  |  |  |                 yield from self.iter_df_rows(idx) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |     def find_section( | 
					
						
							|  |  |  |         self, | 
					
						
							|  |  |  |         section: str, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-21 14:14:26 +00:00
										 |  |  |     ) -> QModelIndex | None: | 
					
						
							| 
									
										
										
										
											2022-02-11 13:32:28 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Find the *first* depth = 1 section matching ``section`` in | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |         the tree and return its index. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         for idx, item in self.iter_d1(): | 
					
						
							|  |  |  |             if item.text() == section: | 
					
						
							|  |  |  |                 return idx | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             # caller must expect his | 
					
						
							|  |  |  |             return None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def clear_section( | 
					
						
							|  |  |  |         self, | 
					
						
							|  |  |  |         section: str, | 
					
						
							|  |  |  |         status_field: str = None, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  |     ) -> None: | 
					
						
							| 
									
										
										
										
											2022-09-12 13:58:11 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Clear all result-rows from under the depth = 1 section. | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         idx = self.find_section(section) | 
					
						
							| 
									
										
										
										
											2021-05-15 23:35:52 +00:00
										 |  |  |         model = self.model() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |         if idx is not None: | 
					
						
							|  |  |  |             if model.hasChildren(idx): | 
					
						
							|  |  |  |                 rows = model.rowCount(idx) | 
					
						
							|  |  |  |                 # print(f'removing {rows} from {section}') | 
					
						
							|  |  |  |                 assert model.removeRows(0, rows, parent=idx) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  |             # remove section as well ? | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |             # model.removeRow(i, QModelIndex()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |             if status_field is not None: | 
					
						
							|  |  |  |                 model.setItem(idx.row(), 1, QStandardItem(status_field)) | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |             else: | 
					
						
							|  |  |  |                 model.setItem(idx.row(), 1, QStandardItem()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |             return idx | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return None | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def set_section_entries( | 
					
						
							|  |  |  |         self, | 
					
						
							|  |  |  |         section: str, | 
					
						
							|  |  |  |         values: Sequence[str], | 
					
						
							|  |  |  |         clear_all: bool = False, | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |         reverse: bool = False, | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ) -> None: | 
					
						
							| 
									
										
										
										
											2022-01-09 16:46:25 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Set result-rows for depth = 1 tree section ``section``. | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         '''
 | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |         if ( | 
					
						
							|  |  |  |             values | 
					
						
							|  |  |  |             and not isinstance(values[0], str) | 
					
						
							|  |  |  |         ): | 
					
						
							|  |  |  |             flattened: list[str] = [] | 
					
						
							|  |  |  |             for val in values: | 
					
						
							|  |  |  |                 flattened.extend(val) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             values = flattened | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if reverse: | 
					
						
							|  |  |  |             values = reversed(values) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |         model = self.model() | 
					
						
							|  |  |  |         if clear_all: | 
					
						
							|  |  |  |             # XXX: rewrite the model from scratch if caller requests it | 
					
						
							|  |  |  |             model.clear() | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         model.setHorizontalHeaderLabels(self.labels) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |         section_idx = self.clear_section(section) | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |         # if we can't find a section start adding to the root | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |         if section_idx is None: | 
					
						
							|  |  |  |             root = model.invisibleRootItem() | 
					
						
							|  |  |  |             section_item = QStandardItem(section) | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |             blank = QStandardItem('') | 
					
						
							|  |  |  |             root.appendRow([section_item, blank]) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |         else: | 
					
						
							|  |  |  |             section_item = model.itemFromIndex(section_idx) | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |         # values just needs to be sequence-like | 
					
						
							|  |  |  |         for i, s in enumerate(values): | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |             ix = QStandardItem(str(i)) | 
					
						
							|  |  |  |             item = QStandardItem(s) | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |             # Add the item to the model | 
					
						
							|  |  |  |             section_item.appendRow([ix, item]) | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |         self.expandAll() | 
					
						
							| 
									
										
										
										
											2021-05-15 23:35:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  |         # TODO: figure out if we can avoid this line in a better way | 
					
						
							|  |  |  |         # such that "re-selection" doesn't happen tree-wise for each new | 
					
						
							|  |  |  |         # sub-search: | 
					
						
							|  |  |  |         # https://doc.qt.io/qt-5/model-view-programming.html#handling-selections-in-item-views | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |         # XXX: THE BELOW LINE MUST BE CALLED. | 
					
						
							|  |  |  |         # this stuff is super finicky and if not done right will cause | 
					
						
							|  |  |  |         # Qt crashes out our buttz. it's required in order to get the | 
					
						
							|  |  |  |         # view to show right after typing input. | 
					
						
							| 
									
										
										
										
											2021-05-28 13:41:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  |         self.select_first() | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-28 13:41:24 +00:00
										 |  |  |         # TODO: the way we might be able to do this more sanely is, | 
					
						
							|  |  |  |         # 1. for the currently selected item, when start rewriting | 
					
						
							|  |  |  |         #    a section figure out the row, column, parent "abstract" | 
					
						
							|  |  |  |         #    position in the tree view and store it | 
					
						
							|  |  |  |         # 2. take that position and re-apply the selection to the new | 
					
						
							|  |  |  |         #    model/tree by looking up the new "equivalent element" and | 
					
						
							|  |  |  |         #    selecting | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 14:20:23 +00:00
										 |  |  |         self.show_matches() | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |     def show_matches( | 
					
						
							|  |  |  |         self, | 
					
						
							| 
									
										
										
										
											2023-02-21 14:14:26 +00:00
										 |  |  |         wh: tuple[float, float] | None = None, | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ) -> None: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if wh: | 
					
						
							|  |  |  |             self.resize_to_results(*wh) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             # case where it's just an update from results and *NOT* | 
					
						
							|  |  |  |             # a resize of some higher level parent-container widget. | 
					
						
							|  |  |  |             search = self.parent() | 
					
						
							|  |  |  |             w, h = search.space_dims() | 
					
						
							| 
									
										
										
										
											2022-09-12 13:58:11 +00:00
										 |  |  |             self.resize_to_results(w=w, h=h) | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |         self.show() | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-23 14:10:59 +00:00
										 |  |  | class SearchBar(Edit): | 
					
						
							| 
									
										
										
										
											2021-07-23 16:00:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-24 20:04:58 +00:00
										 |  |  |     mode_name: str = 'search' | 
					
						
							| 
									
										
										
										
											2021-07-23 16:00:48 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def __init__( | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self, | 
					
						
							|  |  |  |         parent: QWidget, | 
					
						
							| 
									
										
										
										
											2021-07-24 20:04:58 +00:00
										 |  |  |         godwidget: QWidget, | 
					
						
							| 
									
										
										
										
											2023-02-21 14:14:26 +00:00
										 |  |  |         view: CompleterView | None = None, | 
					
						
							| 
									
										
										
										
											2021-07-23 16:00:48 +00:00
										 |  |  |         **kwargs, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ) -> None: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-24 20:04:58 +00:00
										 |  |  |         self.godwidget = godwidget | 
					
						
							| 
									
										
										
										
											2021-07-23 16:00:48 +00:00
										 |  |  |         super().__init__(parent, **kwargs) | 
					
						
							|  |  |  |         self.view: CompleterView = view | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  |     def unfocus(self) -> None: | 
					
						
							| 
									
										
										
										
											2021-05-25 21:10:46 +00:00
										 |  |  |         self.parent().hide() | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  |         self.clearFocus() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |     def hide(self) -> None: | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  |         if self.view: | 
					
						
							|  |  |  |             self.view.hide() | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |         super().hide() | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-05 14:10:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-21 17:53:40 +00:00
										 |  |  | class SearchWidget(QtWidgets.QWidget): | 
					
						
							| 
									
										
										
										
											2022-01-09 16:46:25 +00:00
										 |  |  |     '''
 | 
					
						
							|  |  |  |     Composed widget of ``SearchBar`` + ``CompleterView``. | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     Includes helper methods for item management in the sub-widgets. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     '''
 | 
					
						
							| 
									
										
										
										
											2021-07-24 20:04:58 +00:00
										 |  |  |     mode_name: str = 'search' | 
					
						
							| 
									
										
										
										
											2021-05-30 12:45:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  |     def __init__( | 
					
						
							|  |  |  |         self, | 
					
						
							| 
									
										
										
										
											2021-06-15 22:19:59 +00:00
										 |  |  |         godwidget: 'GodWidget',  # type: ignore # noqa | 
					
						
							| 
									
										
										
										
											2021-08-03 13:47:29 +00:00
										 |  |  |         columns: list[str] = ['src', 'symbol'], | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  |         parent=None, | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ) -> None: | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |         super().__init__(parent) | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # size it as we specify | 
					
						
							|  |  |  |         self.setSizePolicy( | 
					
						
							| 
									
										
										
										
											2024-05-01 18:33:10 +00:00
										 |  |  |             size_policy.Fixed, | 
					
						
							|  |  |  |             size_policy.Fixed, | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-15 22:19:59 +00:00
										 |  |  |         self.godwidget = godwidget | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |         godwidget.reg_for_resize(self) | 
					
						
							| 
									
										
										
										
											2021-05-25 21:10:46 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-21 19:50:09 +00:00
										 |  |  |         self.vbox = QtWidgets.QVBoxLayout(self) | 
					
						
							| 
									
										
										
										
											2021-07-30 14:52:21 +00:00
										 |  |  |         self.vbox.setContentsMargins(0, 4, 4, 0) | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  |         self.vbox.setSpacing(4) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-25 21:10:46 +00:00
										 |  |  |         # split layout for the (label:| search bar entry) | 
					
						
							| 
									
										
										
										
											2021-07-21 19:50:09 +00:00
										 |  |  |         self.bar_hbox = QtWidgets.QHBoxLayout() | 
					
						
							| 
									
										
										
										
											2021-05-25 21:10:46 +00:00
										 |  |  |         self.bar_hbox.setContentsMargins(0, 0, 0, 0) | 
					
						
							|  |  |  |         self.bar_hbox.setSpacing(4) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  |         # add label to left of search bar | 
					
						
							| 
									
										
										
										
											2021-07-21 19:50:09 +00:00
										 |  |  |         self.label = label = QtWidgets.QLabel(parent=self) | 
					
						
							| 
									
										
										
										
											2021-07-26 15:33:37 +00:00
										 |  |  |         label.setStyleSheet( | 
					
						
							|  |  |  |             f"""QLabel {{
 | 
					
						
							|  |  |  |                 color : {hcolor('default_lightest')}; | 
					
						
							|  |  |  |                 font-size : {_font.px_size - 2}px; | 
					
						
							|  |  |  |             }} | 
					
						
							|  |  |  |             """
 | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2024-05-01 18:33:10 +00:00
										 |  |  |         label.setTextFormat( | 
					
						
							|  |  |  |             Qt.TextFormat.MarkdownText | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-05-25 21:10:46 +00:00
										 |  |  |         label.setFont(_font.font) | 
					
						
							|  |  |  |         label.setMargin(4) | 
					
						
							| 
									
										
										
										
											2021-07-26 15:33:37 +00:00
										 |  |  |         label.setText("search:") | 
					
						
							| 
									
										
										
										
											2021-05-25 21:10:46 +00:00
										 |  |  |         label.show() | 
					
						
							|  |  |  |         label.setAlignment( | 
					
						
							| 
									
										
										
										
											2024-05-01 18:33:10 +00:00
										 |  |  |             align_flag.AlignVCenter | 
					
						
							|  |  |  |             | align_flag.AlignLeft | 
					
						
							| 
									
										
										
										
											2021-05-25 21:10:46 +00:00
										 |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.bar_hbox.addWidget(label) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  |         self.view = CompleterView( | 
					
						
							|  |  |  |             parent=self, | 
					
						
							|  |  |  |             labels=columns, | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         self.bar = SearchBar( | 
					
						
							|  |  |  |             parent=self, | 
					
						
							|  |  |  |             view=self.view, | 
					
						
							| 
									
										
										
										
											2021-07-24 20:04:58 +00:00
										 |  |  |             godwidget=godwidget, | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-05-25 21:10:46 +00:00
										 |  |  |         self.bar_hbox.addWidget(self.bar) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         self.vbox.addLayout(self.bar_hbox) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2024-05-01 18:33:10 +00:00
										 |  |  |         self.vbox.setAlignment( | 
					
						
							|  |  |  |             self.bar, | 
					
						
							|  |  |  |             align_flag.AlignTop | 
					
						
							|  |  |  |             | align_flag.AlignRight, | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  |         self.vbox.addWidget(self.bar.view) | 
					
						
							| 
									
										
										
										
											2024-05-01 18:33:10 +00:00
										 |  |  |         self.vbox.setAlignment( | 
					
						
							|  |  |  |             self.view, | 
					
						
							|  |  |  |             align_flag.AlignTop | 
					
						
							|  |  |  |             | align_flag.AlignLeft, | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     def focus(self) -> None: | 
					
						
							| 
									
										
										
										
											2021-05-25 21:10:46 +00:00
										 |  |  |         self.show() | 
					
						
							| 
									
										
										
										
											2022-09-07 21:50:10 +00:00
										 |  |  |         self.bar.focus() | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |     def show_cache_entries( | 
					
						
							|  |  |  |         self, | 
					
						
							|  |  |  |         only: bool = False, | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |         keep_current_item_selected: bool = False, | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |     ) -> None: | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Clear the search results view and show only cached (aka recently | 
					
						
							|  |  |  |         loaded with active data) feeds in the results section. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         godw = self.godwidget | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # first entry in the cache is the current symbol(s) | 
					
						
							| 
									
										
										
										
											2023-05-22 16:13:00 +00:00
										 |  |  |         fqmes = set() | 
					
						
							|  |  |  |         for multi_fqmes in list(godw._chart_cache): | 
					
						
							|  |  |  |             for fqme in set(multi_fqmes): | 
					
						
							|  |  |  |                 fqmes.add(fqme) | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |         if keep_current_item_selected: | 
					
						
							|  |  |  |             sel = self.view.selectionModel() | 
					
						
							|  |  |  |             cidx = sel.currentIndex() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |         self.view.set_section_entries( | 
					
						
							|  |  |  |             'cache', | 
					
						
							| 
									
										
										
										
											2023-05-22 16:13:00 +00:00
										 |  |  |             list(fqmes), | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |             # remove all other completion results except for cache | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |             clear_all=only, | 
					
						
							|  |  |  |             reverse=True, | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |         if ( | 
					
						
							|  |  |  |             keep_current_item_selected | 
					
						
							|  |  |  |             and cidx.isValid() | 
					
						
							|  |  |  |         ): | 
					
						
							|  |  |  |             # set current selection back to what it was before filling out | 
					
						
							|  |  |  |             # the view results. | 
					
						
							|  |  |  |             self.view.select_from_idx(cidx) | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             self.view.select_first() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def get_current_item(self) -> tuple[QModelIndex, str, str] | None: | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Return the current completer tree selection as | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  |         a tuple ``(parent: str, child: str)`` if valid, else ``None``. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         model = self.view.model() | 
					
						
							|  |  |  |         sel = self.view.selectionModel() | 
					
						
							|  |  |  |         cidx = sel.currentIndex() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # TODO: get rid of this hard coded column -> 1 | 
					
						
							|  |  |  |         # and use the ``CompleterView`` schema/settings | 
					
						
							|  |  |  |         # to figure out the desired field(s) | 
					
						
							|  |  |  |         # https://doc.qt.io/qt-5/qstandarditemmodel.html#itemFromIndex | 
					
						
							|  |  |  |         node = model.itemFromIndex(cidx.siblingAtColumn(1)) | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  |         if node: | 
					
						
							|  |  |  |             symbol = node.text() | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |             try: | 
					
						
							|  |  |  |                 provider = node.parent().text() | 
					
						
							|  |  |  |             except AttributeError: | 
					
						
							|  |  |  |                 # no text set | 
					
						
							|  |  |  |                 return None | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # TODO: move this to somewhere non-search machinery specific? | 
					
						
							|  |  |  |             if provider == 'cache': | 
					
						
							|  |  |  |                 symbol, _, provider = symbol.rpartition('.') | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |             return ( | 
					
						
							|  |  |  |                 cidx, | 
					
						
							|  |  |  |                 provider, | 
					
						
							|  |  |  |                 symbol, | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         else: | 
					
						
							|  |  |  |             return None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-13 16:04:54 +00:00
										 |  |  |     async def chart_current_item( | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  |         self, | 
					
						
							|  |  |  |         clear_to_cache: bool = True, | 
					
						
							| 
									
										
										
										
											2021-07-23 16:00:48 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-02-21 14:14:26 +00:00
										 |  |  |     ) -> str | None: | 
					
						
							| 
									
										
										
										
											2022-09-07 21:50:10 +00:00
										 |  |  |         '''
 | 
					
						
							|  |  |  |         Attempt to load and switch the current selected | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  |         completion result to the affiliated chart app. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-07-23 16:00:48 +00:00
										 |  |  |         Return any loaded symbol. | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         value = self.get_current_item() | 
					
						
							|  |  |  |         if value is None: | 
					
						
							|  |  |  |             return None | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |         cidx, provider, symbol = value | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |         godw = self.godwidget | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-05-22 16:13:00 +00:00
										 |  |  |         fqme = f'{symbol}.{provider}' | 
					
						
							|  |  |  |         log.info(f'Requesting symbol: {fqme}') | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |         # assert provider in symbol | 
					
						
							| 
									
										
										
										
											2022-11-07 20:39:28 +00:00
										 |  |  |         await godw.load_symbols( | 
					
						
							| 
									
										
										
										
											2023-05-22 16:13:00 +00:00
										 |  |  |             fqmes=[fqme], | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |             loglevel='info', | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  |         ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # fully qualified symbol name (SNS i guess is what we're | 
					
						
							|  |  |  |         # making?) | 
					
						
							| 
									
										
										
										
											2023-05-22 16:13:00 +00:00
										 |  |  |         fqme = '.'.join([symbol, provider]).lower() | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if clear_to_cache: | 
					
						
							| 
									
										
										
										
											2021-05-30 12:45:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  |             self.bar.clear() | 
					
						
							| 
									
										
										
										
											2021-05-30 12:45:40 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # Re-order the symbol cache on the chart to display in | 
					
						
							|  |  |  |             # LIFO order. this is normally only done internally by | 
					
						
							|  |  |  |             # the chart on new symbols being loaded into memory | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |             godw.set_chart_symbols( | 
					
						
							| 
									
										
										
										
											2023-05-22 16:13:00 +00:00
										 |  |  |                 (fqme,), ( | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |                     godw.hist_linked, | 
					
						
							|  |  |  |                     godw.rt_linked, | 
					
						
							| 
									
										
										
										
											2022-08-30 23:09:18 +00:00
										 |  |  |                 ) | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |             self.show_cache_entries( | 
					
						
							|  |  |  |                 only=True, | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2021-05-30 12:45:40 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-07 21:50:10 +00:00
										 |  |  |         self.bar.focus() | 
					
						
							| 
									
										
										
										
											2023-05-22 16:13:00 +00:00
										 |  |  |         return fqme | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |     def space_dims(self) -> tuple[float, float]: | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Compute and return the "available space dimentions" for this | 
					
						
							|  |  |  |         search widget in terms of px space for results by return the | 
					
						
							|  |  |  |         pair of width and height. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         # XXX: dun need dis rite? | 
					
						
							|  |  |  |         # win = self.window() | 
					
						
							|  |  |  |         # win_h = win.height() | 
					
						
							|  |  |  |         # sb_h = win.statusBar().height() | 
					
						
							|  |  |  |         godw = self.godwidget | 
					
						
							|  |  |  |         hl = godw.hist_linked | 
					
						
							|  |  |  |         edit_h = self.bar.height() | 
					
						
							|  |  |  |         h = hl.height() - edit_h | 
					
						
							|  |  |  |         w = hl.width() | 
					
						
							|  |  |  |         return w, h | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     def on_resize(self) -> None: | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         Resize relay event from god, resize all child widgets. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Right now this is just view to contents and/or the fast chart | 
					
						
							|  |  |  |         height. | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         '''
 | 
					
						
							|  |  |  |         w, h = self.space_dims() | 
					
						
							| 
									
										
										
										
											2022-09-12 17:45:48 +00:00
										 |  |  |         self.bar.view.show_matches(wh=(w, h)) | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:15:51 +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-26 20:50:42 +00:00
										 |  |  | async def pack_matches( | 
					
						
							| 
									
										
										
										
											2021-06-15 22:19:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |     view: CompleterView, | 
					
						
							|  |  |  |     has_results: dict[str, set[str]], | 
					
						
							| 
									
										
										
										
											2021-08-03 13:47:29 +00:00
										 |  |  |     matches: dict[(str, str), list[str]], | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |     provider: str, | 
					
						
							|  |  |  |     pattern: str, | 
					
						
							|  |  |  |     search: Callable[..., Awaitable[dict]], | 
					
						
							| 
									
										
										
										
											2021-08-03 13:47:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |     task_status: TaskStatus[ | 
					
						
							|  |  |  |         trio.CancelScope] = trio.TASK_STATUS_IGNORED, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | ) -> None: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     log.info(f'Searching {provider} for "{pattern}"') | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |     if provider != 'cache': | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  |         # insert provider entries with search status | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |         view.set_section_entries( | 
					
						
							|  |  |  |             section=provider, | 
					
						
							|  |  |  |             values=[], | 
					
						
							|  |  |  |         ) | 
					
						
							|  |  |  |         view.clear_section(provider, status_field='-> searchin..') | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     else:  # for the cache just clear it's entries and don't put a status | 
					
						
							|  |  |  |         view.clear_section(provider) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     with trio.CancelScope() as cs: | 
					
						
							|  |  |  |         task_status.started(cs) | 
					
						
							|  |  |  |         # ensure ^ status is updated | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |         results = list(await search(pattern)) | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |     # XXX: don't cache the cache results xD | 
					
						
							|  |  |  |     if provider != 'cache': | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |         matches[(provider, pattern)] = results | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         # print(f'results from {provider}: {results}') | 
					
						
							|  |  |  |         has_results[pattern].add(provider) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if results: | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  |         # display completion results | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |         view.set_section_entries( | 
					
						
							|  |  |  |             section=provider, | 
					
						
							|  |  |  |             values=results, | 
					
						
							|  |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-05-28 18:08:24 +00:00
										 |  |  |     else: | 
					
						
							|  |  |  |         view.clear_section(provider) | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +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-05 14:10:02 +00:00
										 |  |  |     recv_chan: trio.abc.ReceiveChannel, | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  |     # kb debouncing pauses (bracket defaults) | 
					
						
							| 
									
										
										
										
											2021-06-04 15:29:44 +00:00
										 |  |  |     min_pause_time: float = 0.01,  # absolute min typing throttle | 
					
						
							| 
									
										
										
										
											2021-06-10 15:56:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     # max pause required before slow relay | 
					
						
							|  |  |  |     max_pause_time: float = 6/16 + 0.001, | 
					
						
							| 
									
										
										
										
											2021-05-11 21:26:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-05 14:10:02 +00:00
										 |  |  | ) -> None: | 
					
						
							| 
									
										
										
										
											2022-09-07 21:50:10 +00:00
										 |  |  |     '''
 | 
					
						
							|  |  |  |     Task to search through providers and fill in possible | 
					
						
							| 
									
										
										
										
											2021-05-10 14:10:53 +00:00
										 |  |  |     completion results. | 
					
						
							| 
									
										
										
										
											2021-05-05 14:10:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-07 21:50:10 +00:00
										 |  |  |     '''
 | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |     global _search_active, _search_enabled, _searcher_cache | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 11:51:42 +00:00
										 |  |  |     bar = search.bar | 
					
						
							|  |  |  |     view = bar.view | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |     view.select_from_idx(QModelIndex()) | 
					
						
							| 
									
										
										
										
											2021-05-05 14:10:02 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 11:51:42 +00:00
										 |  |  |     last_text = bar.text() | 
					
						
							| 
									
										
										
										
											2021-05-11 21:26:27 +00:00
										 |  |  |     repeats = 0 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |     # cache of prior patterns to search results | 
					
						
							|  |  |  |     matches = defaultdict(list) | 
					
						
							|  |  |  |     has_results: defaultdict[str, set[str]] = defaultdict(set) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |     # show cached feed list at startup | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |     search.show_cache_entries() | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |     search.on_resize() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-11 21:26:27 +00:00
										 |  |  |     while True: | 
					
						
							|  |  |  |         await _search_active.wait() | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  |         period = None | 
					
						
							| 
									
										
										
										
											2021-05-11 21:26:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  |         while True: | 
					
						
							| 
									
										
										
										
											2021-05-11 21:26:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  |             last_text = bar.text() | 
					
						
							|  |  |  |             wait_start = time.time() | 
					
						
							| 
									
										
										
										
											2021-05-11 21:26:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  |             with trio.move_on_after(max_pause_time): | 
					
						
							|  |  |  |                 pattern = await recv_chan.receive() | 
					
						
							| 
									
										
										
										
											2021-05-11 21:26:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  |             period = time.time() - wait_start | 
					
						
							| 
									
										
										
										
											2022-09-12 19:37:16 +00:00
										 |  |  |             log.debug(f'{pattern} after {period}') | 
					
						
							| 
									
										
										
										
											2021-05-11 21:26:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  |             # during fast multiple key inputs, wait until a pause | 
					
						
							|  |  |  |             # (in typing) to initiate search | 
					
						
							|  |  |  |             if period < min_pause_time: | 
					
						
							|  |  |  |                 log.debug(f'Ignoring fast input for {pattern}') | 
					
						
							|  |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  |             text = bar.text() | 
					
						
							| 
									
										
										
										
											2021-05-27 03:47:20 +00:00
										 |  |  |             # print(f'search: {text}') | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-28 17:44:30 +00:00
										 |  |  |             if not text or text.isspace(): | 
					
						
							| 
									
										
										
										
											2021-05-27 03:47:20 +00:00
										 |  |  |                 # print('idling') | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  |                 _search_active = trio.Event() | 
					
						
							|  |  |  |                 break | 
					
						
							| 
									
										
										
										
											2021-05-11 21:26:27 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  |             if text == last_text: | 
					
						
							|  |  |  |                 repeats += 1 | 
					
						
							| 
									
										
										
										
											2021-05-10 14:10:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  |             if not _search_enabled: | 
					
						
							| 
									
										
										
										
											2021-05-27 03:47:20 +00:00
										 |  |  |                 # print('search currently disabled') | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  |                 break | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |             already_has_results = has_results[text] | 
					
						
							| 
									
										
										
										
											2021-06-10 15:56:41 +00:00
										 |  |  |             log.debug(f'Search req for {text}') | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # issue multi-provider fan-out search request and place | 
					
						
							|  |  |  |             # "searching.." statuses on outstanding results providers | 
					
						
							|  |  |  |             async with trio.open_nursery() as n: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-27 15:40:08 +00:00
										 |  |  |                 for provider, (search, pause) in ( | 
					
						
							|  |  |  |                     _searcher_cache.copy().items() | 
					
						
							|  |  |  |                 ): | 
					
						
							| 
									
										
										
										
											2021-06-04 15:29:44 +00:00
										 |  |  |                     # XXX: only conduct search on this backend if it's | 
					
						
							| 
									
										
										
										
											2021-06-10 15:56:41 +00:00
										 |  |  |                     # registered for the corresponding pause period AND | 
					
						
							|  |  |  |                     # it hasn't already been searched with the current | 
					
						
							|  |  |  |                     # input pattern (in which case just look up the old | 
					
						
							|  |  |  |                     # results). | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |                     if ( | 
					
						
							|  |  |  |                         period >= pause | 
					
						
							|  |  |  |                         and provider not in already_has_results | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |                     ): | 
					
						
							| 
									
										
										
										
											2021-06-10 15:56:41 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                         # TODO: it may make more sense TO NOT search the | 
					
						
							|  |  |  |                         # cache in a bg task since we know it's fully | 
					
						
							|  |  |  |                         # cpu-bound. | 
					
						
							|  |  |  |                         if provider != 'cache': | 
					
						
							|  |  |  |                             view.clear_section( | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |                                 provider, | 
					
						
							|  |  |  |                                 status_field='-> searchin..', | 
					
						
							|  |  |  |                             ) | 
					
						
							| 
									
										
										
										
											2021-06-10 15:56:41 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |                         await n.start( | 
					
						
							|  |  |  |                             pack_matches, | 
					
						
							|  |  |  |                             view, | 
					
						
							|  |  |  |                             has_results, | 
					
						
							|  |  |  |                             matches, | 
					
						
							|  |  |  |                             provider, | 
					
						
							|  |  |  |                             text, | 
					
						
							|  |  |  |                             search | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |                         ) | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |                     else:  # already has results for this input text | 
					
						
							|  |  |  |                         results = matches[(provider, text)] | 
					
						
							| 
									
										
										
										
											2021-06-04 15:29:44 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                         # TODO really for the cache we need an | 
					
						
							|  |  |  |                         # invalidation signal so that we only re-search | 
					
						
							|  |  |  |                         # the cache once it's been mutated by the chart | 
					
						
							|  |  |  |                         # switcher.. right now we're just always | 
					
						
							|  |  |  |                         # re-searching it's ``dict`` since it's easier | 
					
						
							|  |  |  |                         # but it also causes it to be slower then cached | 
					
						
							|  |  |  |                         # results from other providers on occasion. | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |                         if ( | 
					
						
							|  |  |  |                             results | 
					
						
							|  |  |  |                         ): | 
					
						
							|  |  |  |                             if provider != 'cache': | 
					
						
							|  |  |  |                                 view.set_section_entries( | 
					
						
							|  |  |  |                                     section=provider, | 
					
						
							|  |  |  |                                     values=results, | 
					
						
							|  |  |  |                                 ) | 
					
						
							|  |  |  |                             else: | 
					
						
							|  |  |  |                                 # if provider == 'cache': | 
					
						
							|  |  |  |                                 # for the cache just show what we got | 
					
						
							|  |  |  |                                 # that matches | 
					
						
							|  |  |  |                                 search.show_cache_entries() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 20:50:42 +00:00
										 |  |  |                         else: | 
					
						
							|  |  |  |                             view.clear_section(provider) | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-10 15:56:41 +00:00
										 |  |  |             if repeats > 2 and period > max_pause_time: | 
					
						
							|  |  |  |                 _search_active = trio.Event() | 
					
						
							|  |  |  |                 repeats = 0 | 
					
						
							|  |  |  |                 break | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  |             bar.show() | 
					
						
							| 
									
										
										
										
											2021-05-15 23:35:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-14 11:51:42 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  | async def handle_keyboard_input( | 
					
						
							| 
									
										
										
										
											2021-05-10 14:10:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-15 22:19:59 +00:00
										 |  |  |     searchbar: SearchBar, | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  |     recv_chan: trio.abc.ReceiveChannel, | 
					
						
							| 
									
										
										
										
											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 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |     searchw = searchbar.parent() | 
					
						
							|  |  |  |     godwidget = searchw.godwidget | 
					
						
							|  |  |  |     view = searchbar.view | 
					
						
							|  |  |  |     view.set_font_size(searchbar.dpi_font.px_size) | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |     send, recv = trio.open_memory_channel(616) | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     async with trio.open_nursery() as n: | 
					
						
							| 
									
										
										
										
											2021-05-26 17:49:14 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         # start a background multi-searcher task which receives | 
					
						
							|  |  |  |         # patterns relayed from this keyboard input handler and | 
					
						
							|  |  |  |         # async updates the completer view's results. | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  |         n.start_soon( | 
					
						
							| 
									
										
										
										
											2021-05-14 11:51:42 +00:00
										 |  |  |             partial( | 
					
						
							|  |  |  |                 fill_results, | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |                 searchw, | 
					
						
							| 
									
										
										
										
											2021-05-14 11:51:42 +00:00
										 |  |  |                 recv, | 
					
						
							|  |  |  |             ) | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  |         ) | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |         searchbar.focus() | 
					
						
							|  |  |  |         searchw.show_cache_entries() | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |         await trio.sleep(0) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-10 20:57:19 +00:00
										 |  |  |         async for kbmsg in recv_chan: | 
					
						
							|  |  |  |             event, etype, key, mods, txt = kbmsg.to_tuple() | 
					
						
							| 
									
										
										
										
											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-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 18:28:08 +00:00
										 |  |  |             ctl = False | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  |             if mods == Qt.ControlModifier: | 
					
						
							| 
									
										
										
										
											2021-05-20 18:28:08 +00:00
										 |  |  |                 ctl = True | 
					
						
							| 
									
										
										
										
											2021-04-23 15:12:29 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |             if key in ( | 
					
						
							|  |  |  |                 Qt.Key_Enter, | 
					
						
							|  |  |  |                 Qt.Key_Return | 
					
						
							|  |  |  |             ): | 
					
						
							| 
									
										
										
										
											2021-05-11 21:26:27 +00:00
										 |  |  |                 _search_enabled = False | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |                 await searchw.chart_current_item(clear_to_cache=True) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 # XXX: causes hang and segfault.. | 
					
						
							|  |  |  |                 # searchw.show_cache_entries( | 
					
						
							|  |  |  |                 #     only=True, | 
					
						
							|  |  |  |                 #     keep_current_item_selected=True, | 
					
						
							|  |  |  |                 # ) | 
					
						
							| 
									
										
										
										
											2021-05-20 18:28:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |                 view.show_matches() | 
					
						
							|  |  |  |                 searchw.focus() | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |             elif ( | 
					
						
							|  |  |  |                 not ctl | 
					
						
							|  |  |  |                 and not searchbar.text() | 
					
						
							|  |  |  |             ): | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |                 # TODO: really should factor this somewhere..bc | 
					
						
							|  |  |  |                 # we're doin it in another spot as well.. | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |                 searchw.show_cache_entries(only=True) | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  |                 continue | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |             # cancel and close | 
					
						
							| 
									
										
										
										
											2021-05-20 18:28:08 +00:00
										 |  |  |             if ctl and key in { | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |                 Qt.Key_C, | 
					
						
							|  |  |  |                 Qt.Key_Space,   # i feel like this is the "native" one | 
					
						
							|  |  |  |                 Qt.Key_Alt, | 
					
						
							|  |  |  |             }: | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |                 searchbar.unfocus() | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |                 # kill the search and focus back on main chart | 
					
						
							| 
									
										
										
										
											2021-08-01 22:53:59 +00:00
										 |  |  |                 if godwidget: | 
					
						
							| 
									
										
										
										
											2021-08-03 13:47:29 +00:00
										 |  |  |                     godwidget.focus() | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |                 continue | 
					
						
							| 
									
										
										
										
											2021-05-06 20:41:15 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |             if ( | 
					
						
							|  |  |  |                 ctl | 
					
						
							|  |  |  |                 and key in {Qt.Key_L} | 
					
						
							|  |  |  |             ): | 
					
						
							| 
									
										
										
										
											2021-05-21 16:44:24 +00:00
										 |  |  |                 # like url (link) highlight in a web browser | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |                 searchbar.focus() | 
					
						
							| 
									
										
										
										
											2021-05-21 16:44:24 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |             # selection navigation controls | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |             elif ( | 
					
						
							|  |  |  |                 ctl | 
					
						
							|  |  |  |                 and key in {Qt.Key_D} | 
					
						
							|  |  |  |             ): | 
					
						
							| 
									
										
										
										
											2021-05-20 18:28:08 +00:00
										 |  |  |                 view.next_section(direction='down') | 
					
						
							| 
									
										
										
										
											2021-05-28 13:41:24 +00:00
										 |  |  |                 _search_enabled = False | 
					
						
							| 
									
										
										
										
											2021-05-20 18:28:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |             elif ( | 
					
						
							|  |  |  |                 ctl | 
					
						
							|  |  |  |                 and key in {Qt.Key_U} | 
					
						
							|  |  |  |             ): | 
					
						
							| 
									
										
										
										
											2021-05-20 18:28:08 +00:00
										 |  |  |                 view.next_section(direction='up') | 
					
						
							| 
									
										
										
										
											2021-05-28 13:41:24 +00:00
										 |  |  |                 _search_enabled = False | 
					
						
							| 
									
										
										
										
											2021-05-20 18:28:08 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |             # selection navigation controls | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |             elif ( | 
					
						
							|  |  |  |                 ctl and ( | 
					
						
							|  |  |  |                     key in { | 
					
						
							|  |  |  |                         Qt.Key_K, | 
					
						
							|  |  |  |                         Qt.Key_J, | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     or key in { | 
					
						
							|  |  |  |                         Qt.Key_Up, | 
					
						
							|  |  |  |                         Qt.Key_Down, | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 ) | 
					
						
							|  |  |  |             ): | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |                 _search_enabled = False | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if key in { | 
					
						
							|  |  |  |                     Qt.Key_K, | 
					
						
							|  |  |  |                     Qt.Key_Up | 
					
						
							|  |  |  |                 }: | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |                     item = view.select_previous() | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |                 elif key in { | 
					
						
							|  |  |  |                     Qt.Key_J, | 
					
						
							|  |  |  |                     Qt.Key_Down, | 
					
						
							|  |  |  |                 }: | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |                     item = view.select_next() | 
					
						
							| 
									
										
										
										
											2021-05-18 15:22:59 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-20 12:13:50 +00:00
										 |  |  |                 if item: | 
					
						
							|  |  |  |                     parent_item = item.parent() | 
					
						
							| 
									
										
										
										
											2021-05-20 18:28:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-10 01:46:57 +00:00
										 |  |  |                     # if we're in the cache section and thus the next | 
					
						
							|  |  |  |                     # selection is a cache item, switch and show it | 
					
						
							|  |  |  |                     # immediately since it should be very fast. | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |                     if ( | 
					
						
							|  |  |  |                         parent_item | 
					
						
							|  |  |  |                         and parent_item.text() == 'cache' | 
					
						
							|  |  |  |                     ): | 
					
						
							|  |  |  |                         await searchw.chart_current_item(clear_to_cache=False) | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |             # ACTUAL SEARCH BLOCK # | 
					
						
							|  |  |  |             # where we fuzzy complete and fill out sections. | 
					
						
							| 
									
										
										
										
											2021-05-20 18:28:08 +00:00
										 |  |  |             elif not ctl: | 
					
						
							| 
									
										
										
										
											2021-05-11 21:26:27 +00:00
										 |  |  |                 # relay to completer task | 
					
						
							|  |  |  |                 _search_enabled = True | 
					
						
							| 
									
										
										
										
											2023-01-09 20:06:12 +00:00
										 |  |  |                 send.send_nowait(searchw.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, | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-03 13:47:29 +00:00
										 |  |  | ) -> dict[str, Any]: | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-10-01 21:46:46 +00:00
										 |  |  |     tokens: list[str] = [] | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |     for key in source: | 
					
						
							| 
									
										
										
										
											2023-10-01 21:46:46 +00:00
										 |  |  |         match key: | 
					
						
							|  |  |  |             case str(): | 
					
						
							|  |  |  |                 tokens.append(key) | 
					
						
							|  |  |  |             case []: | 
					
						
							|  |  |  |                 tokens.extend(key) | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  |     # search routine can be specified as a function such | 
					
						
							|  |  |  |     # as in the case of the current app's local symbol cache | 
					
						
							| 
									
										
										
										
											2023-10-01 21:46:46 +00:00
										 |  |  |     matches = fuzzy.extract( | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  |         text, | 
					
						
							| 
									
										
										
										
											2022-11-15 16:22:08 +00:00
										 |  |  |         tokens, | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  |         score_cutoff=90, | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2023-10-01 21:46:46 +00:00
										 |  |  |     log.info( | 
					
						
							|  |  |  |         'cache search results:\n' | 
					
						
							|  |  |  |         f'{pformat(matches)}' | 
					
						
							|  |  |  |     ) | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  |     return [item[0] for item in matches] | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | # cache of provider names to async search routines | 
					
						
							| 
									
										
										
										
											2021-08-03 13:47:29 +00:00
										 |  |  | _searcher_cache: dict[str, Callable[..., Awaitable]] = {} | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | @asynccontextmanager | 
					
						
							|  |  |  | async def register_symbol_search( | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     provider_name: str, | 
					
						
							|  |  |  |     search_routine: Callable, | 
					
						
							| 
									
										
										
										
											2023-02-21 14:14:26 +00:00
										 |  |  |     pause_period: float | None = None, | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  | ) -> AsyncIterator[dict]: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     global _searcher_cache | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-04 15:29:44 +00:00
										 |  |  |     pause_period = pause_period or 0.1 | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  |     # deliver search func to consumer | 
					
						
							|  |  |  |     try: | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  |         _searcher_cache[provider_name] = (search_routine, pause_period) | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  |         yield search_routine | 
					
						
							| 
									
										
										
										
											2021-05-18 12:19:52 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-05-17 00:52:22 +00:00
										 |  |  |     finally: | 
					
						
							|  |  |  |         _searcher_cache.pop(provider_name) |