diff --git a/BuiltInSearchResults.py b/BuiltInSearchResults.py new file mode 100644 index 0000000..f9bba40 --- /dev/null +++ b/BuiltInSearchResults.py @@ -0,0 +1,33 @@ +# You can add your own result proviers and action/tooltip handlers, by importing this module and calling the registration functions as follows. +# We use wrapper functions which import the actual implementation and call it, in order to avoid loading too much code during startup. + +import SearchResults + +SearchResults.registerResultProvider('refreshTools', + getItemGroupsCached = lambda: __import__('ResultsRefreshTools').refreshToolsResultsProvider(), + getItemGroupsUncached = lambda: []) +SearchResults.registerResultProvider('document', + getItemGroupsCached = lambda: [], + getItemGroupsUncached = lambda: __import__('ResultsDocument').documentResultsProvider()) +SearchResults.registerResultProvider('toolbar', + getItemGroupsCached = lambda: __import__('ResultsToolbar').toolbarResultsProvider(), + getItemGroupsUncached = lambda: []) + +SearchResults.registerResultHandler('refreshTools', + action = lambda nfo: __import__('ResultsRefreshTools').refreshToolsAction(nfo), + toolTip = lambda nfo, setParent: __import__('ResultsRefreshTools').refreshToolsToolTip(nfo, setParent)) +SearchResults.registerResultHandler('toolbar', + action = lambda nfo: __import__('ResultsToolbar').toolbarAction(nfo), + toolTip = lambda nfo, setParent: __import__('ResultsToolbar').toolbarToolTip(nfo, setParent)) +SearchResults.registerResultHandler('tool', + action = lambda nfo : __import__('ResultsToolbar').subToolAction(nfo), + toolTip = lambda nfo, setParent: __import__('ResultsToolbar').subToolToolTip(nfo, setParent)) +SearchResults.registerResultHandler('subtool', + action = lambda nfo : __import__('ResultsToolbar').subToolAction(nfo), + toolTip = lambda nfo, setParent: __import__('ResultsToolbar').subToolToolTip(nfo, setParent)) +SearchResults.registerResultHandler('document', + action = lambda nfo : __import__('ResultsDocument').documentAction(nfo), + toolTip = lambda nfo, setParent: __import__('ResultsDocument').documentToolTip(nfo, setParent)) +SearchResults.registerResultHandler('documentObject', + action = lambda nfo : __import__('ResultsDocument').documentObjectAction(nfo), + toolTip = lambda nfo, setParent: __import__('ResultsDocument').documentObjectToolTip(nfo, setParent)) diff --git a/GatherTools.py b/GatherTools.py deleted file mode 100644 index e69de29..0000000 diff --git a/GetItemGroups.py b/GetItemGroups.py new file mode 100644 index 0000000..42b1bd3 --- /dev/null +++ b/GetItemGroups.py @@ -0,0 +1,57 @@ +globalGroups = [] + +itemGroups = None +serializedItemGroups = None + +def onResultSelected(index, groupId): + global globalGroups + nfo = globalGroups[groupId] + handlerName = nfo['action']['handler'] + import SearchResults + if handlerName in SearchResults.actionHandlers: + SearchResults.actionHandlers[handlerName](nfo) + else: + from PySide import QtGui + QtGui.QMessageBox.warning(None, 'Could not execute this action', 'Could not execute this action, it could be from a Mod that has been uninstalled. Try refreshing the list of tools.') + +def getToolTip(groupId, setParent): + global globalGroups + nfo = globalGroups[int(groupId)] + handlerName = nfo['action']['handler'] + import SearchResults + if handlerName in SearchResults.toolTipHandlers: + return SearchResults.toolTipHandlers[handlerName](nfo, setParent) + else: + return 'Could not load tooltip for this tool, it could be from a Mod that has been uninstalled. Try refreshing the list of tools.' + +def getItemGroups(): + global itemGroups, serializedItemGroups, globalGroups + + # Import the tooltip+action handlers and search result providers that are bundled with this Mod. + # Other providers should import SearchResults and register their handlers and providers + import BuiltInSearchResults + + # Load the list of tools, preferably from the cache, if it has not already been loaded: + if itemGroups is None: + if serializedItemGroups is None: + import RefreshTools + itemGroups = RefreshTools.refreshToolbars(doLoadAllWorkbenches = False) + else: + import Serialize + itemGroups = Serialize.deserialize(serializedItemGroups) + + # Aggregate the tools (cached) and document objects (not cached), and assign an index to each + import SearchResults + igs = itemGroups + for providerName, provider in SearchResults.resultProvidersUncached.items(): + igs = igs + provider() + globalGroups = [] + def addId(group): + globalGroups.append(group) + group['id'] = len(globalGroups) - 1 + for subitem in group['subitems']: + addId(subitem) + for ig in igs: + addId(ig) + + return igs diff --git a/IndentedItemDelegate.py b/IndentedItemDelegate.py new file mode 100644 index 0000000..aff7046 --- /dev/null +++ b/IndentedItemDelegate.py @@ -0,0 +1,12 @@ +from PySide import QtGui + +# Inspired by https://stackoverflow.com/a/5443220/324969 +# Inspired by https://forum.qt.io/topic/69807/qtreeview-indent-entire-row +class IndentedItemDelegate(QtGui.QStyledItemDelegate): + def __init__(self): + super(IndentedItemDelegate, self).__init__() + def paint(self, painter, option, index): + depth = int(option.widget.model().itemData(index.siblingAtColumn(1))[0]) + indent = 16 * depth + option.rect.adjust(indent, 0, 0, 0) + super(IndentedItemDelegate, self).paint(painter, option, index) diff --git a/InitGui.py b/InitGui.py index 55107e2..ce3f9ec 100644 --- a/InitGui.py +++ b/InitGui.py @@ -1 +1,27 @@ -import SearchTools +# Avoid garbage collection by storing the action in a global variable +wax = None + +def addToolSearchBox(): + import FreeCADGui + from PySide import QtGui + import SearchBox + global wax, sea + mw = FreeCADGui.getMainWindow() + mbr = mw.findChildren(QtGui.QToolBar, 'File') + # The toolbar will be unavailable if this file is loaded during startup, because no workbench is active and no toolbars are visible. + if len(mbr) > 0: + # Get the first toolbar named 'File', and add + mbr = mbr[0] + # Create search box widget + sea = SearchBox.SearchBox(getItemGroups = lambda: __import__('GetItemGroups').getItemGroups(), + getToolTip = lambda groupId, setParent: __import__('GetItemGroups').getToolTip(groupId, setParent), + getItemDelegate = lambda: __import__('IndentedItemDelegate').IndentedItemDelegate()) + sea.resultSelected.connect(lambda index, groupId: __import__('GetItemGroups').onResultSelected(index, groupId)) + wax = QtGui.QWidgetAction(None) + wax.setDefaultWidget(sea) + #mbr.addWidget(sea) + mbr.addAction(wax) + +addToolSearchBox() +import FreeCADGui +FreeCADGui.getMainWindow().workbenchActivated.connect(addToolSearchBox) diff --git a/RefreshTools.py b/RefreshTools.py new file mode 100644 index 0000000..7a15a03 --- /dev/null +++ b/RefreshTools.py @@ -0,0 +1,81 @@ +import os +import FreeCAD as App + +def loadAllWorkbenches(): + from PySide import QtGui + import FreeCADGui + activeWorkbench = FreeCADGui.activeWorkbench().name() + lbl = QtGui.QLabel('Loading workbench … (…/…)') + lbl.show() + lst = FreeCADGui.listWorkbenches() + for i, wb in enumerate(lst): + msg = 'Loading workbench ' + wb + ' (' + str(i) + '/' + str(len(lst)) + ')' + print(msg) + lbl.setText(msg) + geo = lbl.geometry() + geo.setSize(lbl.sizeHint()) + lbl.setGeometry(geo) + lbl.repaint() + FreeCADGui.updateGui() # Probably slower with this, because it redraws the entire GUI with all tool buttons changed etc. but allows the label to actually be updated, and it looks nice and gives a quick overview of all the workbenches… + try: + FreeCADGui.activateWorkbench(wb) + except: + pass + lbl.hide() + FreeCADGui.activateWorkbench(activeWorkbench) + +def cachePath(): + return os.path.join(App.getUserAppDataDir(), 'Cache_SearchToolsMod') + +def gatherTools(): + itemGroups = [] + import SearchResults + for providerName, provider in SearchResults.resultProvidersCached.items(): + itemGroups = itemGroups + provider() + return itemGroups + +def writeCacheTools(): + import Serialize + serializedItemGroups = Serialize.serialize(gatherTools()) + # Todo: use wb and a specific encoding. + with open(cachePath(), 'w') as cache: + cache.write(serializedItemGroups) + # I prefer to systematically deserialize, instead of taking the original version, + # this avoids possible inconsistencies between the original and the cache and + # makes sure cache-related bugs are noticed quickly. + import Serialize + itemGroups = Serialize.deserialize(serializedItemGroups) + print('SearchBox: Cache has been written.') + return itemGroups + +def readCacheTools(): + # Todo: use rb and a specific encoding. + with open(cachePath(), 'r') as cache: + serializedItemGroups = cache.read() + import Serialize + itemGroups = Serialize.deserialize(serializedItemGroups) + print('SearchBox: Tools were loaded from the cache.') + return itemGroups + + +def refreshToolbars(doLoadAllWorkbenches = True): + if doLoadAllWorkbenches: + loadAllWorkbenches() + return writeCacheTools() + else: + try: + return readCacheTools() + except: + return writeCacheTools() + +def refreshToolsAction(): + from PySide import QtGui + print('Refresh list of tools') + fw = QtGui.QApplication.focusWidget() + if fw is not None: + fw.clearFocus() + reply = QtGui.QMessageBox.question(None, "Load all workbenches?", "Load all workbenches? This can cause FreeCAD to become unstable, and this \"reload tools\" feature contained a bug that crashed freecad systematically, so please make sure you save your work before. It's a good idea to restart FreeCAD after this operation.", QtGui.QMessageBox.Yes, QtGui.QMessageBox.No) + if reply == QtGui.QMessageBox.Yes: + refreshToolbars() + else: + print('cancelled') diff --git a/ResultsDocument.py b/ResultsDocument.py new file mode 100644 index 0000000..0050c28 --- /dev/null +++ b/ResultsDocument.py @@ -0,0 +1,118 @@ +from PySide import QtGui +from PySide import QtCore +import FreeCAD as App +import FreeCADGui +import SafeViewer +import SearchBox + +def documentAction(nfo): + act = nfo['action'] + # Todo: this should also select the document in the tree view + print('switch to document ' + act['document']) + App.setActiveDocument(act['document']) + +def documentObjectAction(nfo): + act = nfo['action'] + print('select object ' + act['document'] + '.' + act['object']) + FreeCADGui.Selection.addSelection(act['document'], act['object']) + +# For some reason, the viewer always works except when used for two consecutive items in the search results: it then disappears after a short zoom-in+zoom-out animation. +# I'm giving up on getting this viewer to work in a clean way, and will try swapping two instances so that the same one is never used twice in a row. +# Also, in order to avoid segfaults when the module is reloaded (which causes the previous viewer to be garbage collected at some point), we're using a global property that will survive module reloads. +if not hasattr(App, '_SearchTools3DViewer'): + # Toggle between + App._SearchTools3DViewer = None + App._SearchTools3DViewerB = None + +class DocumentObjectToolTipWidget(QtGui.QWidget): + def __init__(self, nfo, setParent): + import pivy + super(DocumentObjectToolTipWidget, self).__init__() + html = '
' + nfo['toolTip']['label'] + '
App.getDocument(' + repr(str(nfo['toolTip']['docName'])) + ').getObject(' + repr(str(nfo['toolTip']['name'])) + ')
' + nfo['toolTip']['label'] + '
App.getDocument(' + repr(str(nfo['toolTip']['name'])) + ')
Load all workbenches to refresh this list of tools. This may take a minute, depending on the number of installed workbenches.
' + +genericToolIcon = QtGui.QIcon(QtGui.QIcon(os.path.dirname(__file__) + '/Tango-Tools-spanner-hammer.svg')) +def refreshToolsResultsProvider(): + return [ + { + 'icon': genericToolIcon, + 'text': 'Refresh list of tools', + 'toolTip': '', + 'action': {'handler': 'refreshTools'}, + 'subitems': [] + } + ] \ No newline at end of file diff --git a/ResultsToolbar.py b/ResultsToolbar.py new file mode 100644 index 0000000..f2dd122 --- /dev/null +++ b/ResultsToolbar.py @@ -0,0 +1,103 @@ +from PySide import QtGui +import FreeCADGui +import Serialize + +def toolbarAction(nfo): + act = nfo['action'] + print('show toolbar ' + act['toolbar'] + ' from workbenches ' + repr(act['workbenches'])) + +def subToolAction(nfo): + act = nfo['action'] + toolPath = act['toolbar'] + '.' + act['tool'] + if 'subTool' in act: + toolPath = toolPath + '.' + act['subTool'] + def runTool(): + mw = FreeCADGui.getMainWindow() + for the_toolbar in mw.findChildren(QtGui.QToolBar, act['toolbar']): + for tbt in the_toolbar.findChildren(QtGui.QToolButton): + if tbt.text() == act['tool']: + action = None + if 'subTool' in act: + men = tbt.menu() + if men: + for mac in men.actions(): + if mac.text() == act['subTool']: + action = mac + break + else: + action = tbt.defaultAction() + if 'showMenu' in act and act['showMenu']: + print('Popup submenu of tool ' + toolPath + ' available in workbenches ' + repr(act['workbenches'])) + the_toolbar.show() + tbt.showMenu() + return True + elif action is not None: + print('Run action of tool ' + toolPath + ' available in workbenches ' + repr(act['workbenches'])) + action.trigger() + return True + return False + if runTool(): + return + else: + for workbench in act['workbenches']: + print('Activating workbench ' + workbench + ' to access tool ' + toolPath) + FreeCADGui.activateWorkbench(workbench) + if runTool(): + return + print('Tool ' + toolPath + ' not found, was it offered by an extension that is no longer present?') + +def toolbarToolTip(nfo, setParent): + return 'Display toolbar ' + nfo['toolTip'] + '
This toolbar appears in the following workbenches:
' + nfo['toolTip'] + '
' + +def getAllToolbars(): + all_tbs = dict() + for wbname, workbench in FreeCADGui.listWorkbenches().items(): + try: + tbs = workbench.listToolbars() + except: + continue + # careful, tbs contains all the toolbars of the workbench, including shared toolbars + for tb in tbs: + if tb not in all_tbs: + all_tbs[tb] = set() + all_tbs[tb].add(wbname) + return all_tbs + +def toolbarResultsProvider(): + itemGroups = [] + all_tbs = getAllToolbars() + mw = FreeCADGui.getMainWindow() + for toolbarName, toolbarIsInWorkbenches in all_tbs.items(): + toolbarIsInWorkbenches = sorted(list(toolbarIsInWorkbenches)) + for the_toolbar in mw.findChildren(QtGui.QToolBar, toolbarName): + group = [] + for tbt in the_toolbar.findChildren(QtGui.QToolButton): + text = tbt.text() + act = tbt.defaultAction() + if text != '': + # TODO: there also is the tooltip + icon = tbt.icon() + men = tbt.menu() + subgroup = [] + if men: + subgroup = [] + for mac in men.actions(): + if mac.text(): + action = { 'handler': 'subTool', 'workbenches': toolbarIsInWorkbenches, 'toolbar': toolbarName, 'tool': text, 'subTool': mac.text() } + subgroup.append({'icon':mac.icon(), 'text':mac.text(), 'toolTip': mac.toolTip(), 'action':action, 'subitems':[]}) + # The default action of a menu changes dynamically, instead of triggering the last action, just show the menu. + action = { 'handler': 'tool', 'workbenches': toolbarIsInWorkbenches, 'toolbar': toolbarName, 'tool': text, 'showMenu': bool(men) } + group.append({'icon':icon, 'text':text, 'toolTip': tbt.toolTip(), 'action': action, 'subitems': subgroup}) + # TODO: move the 'workbenches' field to the itemgroup + action = { 'handler': 'toolbar', 'workbenches': toolbarIsInWorkbenches, 'toolbar': toolbarName } + itemGroups.append({ + 'icon': QtGui.QIcon(':/icons/Group.svg'), + 'text': toolbarName, + 'toolTip': '', + 'action': action, + 'subitems': group + }) + return itemGroups diff --git a/SafeViewer.py b/SafeViewer.py new file mode 100644 index 0000000..92a2249 --- /dev/null +++ b/SafeViewer.py @@ -0,0 +1,71 @@ +from PySide import QtGui + +class SafeViewer(QtGui.QWidget): + """FreeCAD uses a modified version of QuarterWidget, so the import pivy.quarter one will cause segfaults. + FreeCAD's FreeCADGui.createViewer() puts the viewer widget inside an MDI window, and detaching it without causing segfaults on exit is tricky. + This class contains some kludges to extract the viewer as a standalone widget and destroy it safely.""" + def __init__(self, parent = None): + super(SafeViewer, self).__init__() + import FreeCADGui + self.viewer = FreeCADGui.createViewer() + self.graphicsView = self.viewer.graphicsView() + self.oldGraphicsViewParent = self.graphicsView.parent() + self.oldGraphicsViewParentParent = self.oldGraphicsViewParent.parent() + self.oldGraphicsViewParentParentParent = self.oldGraphicsViewParentParent.parent() + + # Avoid segfault but still hide the undesired window by moving it to a new hidden MDI area. + self.hiddenQMDIArea = QtGui.QMdiArea() + self.hiddenQMDIArea.addSubWindow(self.oldGraphicsViewParentParentParent) + + self.private_widget = self.oldGraphicsViewParent + self.private_widget.setParent(parent) + + self.setLayout(QtGui.QVBoxLayout()) + self.layout().addWidget(self.private_widget) + self.layout().setContentsMargins(0,0,0,0) + + def fin(slf): + slf.finalizer() + + import weakref + weakref.finalize(self, fin, self) + + self.destroyed.connect(self.finalizer) + + def finalizer(self): + # Cleanup in an order that doesn't cause a segfault: + self.private_widget.setParent(self.oldGraphicsViewParentParent) + self.oldGraphicsViewParentParentParent.close() + self.oldGraphicsViewParentParentParent = None + self.oldGraphicsViewParentParent = None + self.oldGraphicsViewParent = None + self.graphicsView = None + self.viewer = None + #self.parent = None + self.hiddenQMDIArea = None + +""" +# Example use: +from PySide import QtGui +import pivy +def mk(v): + w = QtGui.QMainWindow() + oldFocus = QtGui.QApplication.focusWidget() + sv.widget.setParent(w) + oldFocus.setFocus() + w.show() + col = pivy.coin.SoBaseColor() + col.rgb = (1, 0, 0) + trans = pivy.coin.SoTranslation() + trans.translation.setValue([0, 0, 0]) + cub = pivy.coin.SoCube() + myCustomNode = pivy.coin.SoSeparator() + myCustomNode.addChild(col) + myCustomNode.addChild(trans) + myCustomNode.addChild(cub) + sv.viewer.getViewer().setSceneGraph(myCustomNode) + sv.viewer.fitAll() + return w +sv = SafeViewer() +ww=mk(sv) +""" diff --git a/SearchBox.py b/SearchBox.py index 519d7bb..e0ca8a8 100644 --- a/SearchBox.py +++ b/SearchBox.py @@ -1,109 +1,269 @@ -if True: - from PySide import QtGui - class SearchBox(QtGui.QLineEdit): - def __init__(self, model, maxVisibleRows, parent): - # Call parent cosntructor - super(SearchBox, self).__init__(parent) - # Save arguments - self.model = model - self.maxVisibleRows = maxVisibleRows - # Create list view - self.listView = QtGui.QListView(self) - self.listView.setWindowFlags(QtGui.Qt.ToolTip) - self.listView.setWindowFlag(QtGui.Qt.FramelessWindowHint) - self.listView.setModel(self.model) - # make the QListView non-editable - self.listView.setEditTriggers(QtGui.QAbstractItemView.NoEditTriggers) - # Connect signals and slots - self.textChanged.connect(self.filterList) - self.listView.clicked.connect(self.selectResult) - self.listView.selectionModel().selectionChanged.connect(self.showExtraInfo) - def focusInEvent(self, qFocusEvent): - self.showList() - super(SearchBox, self).focusInEvent(qFocusEvent) - def focusOutEvent(self, qFocusEvent): - self.listView.hide() - super(SearchBox, self).focusOutEvent(qFocusEvent) - def keyPressEvent(self, qKeyEvent): - key = qKeyEvent.key() - listMovementKeys = { - QtCore.Qt.Key_Down: lambda current, nbRows: (current + 1) % nbRows, - QtCore.Qt.Key_Up: lambda current, nbRows: (current - 1) % nbRows, - QtCore.Qt.Key_PageDown: lambda current, nbRows: max(current + min(1, self.maxVisibleRows / 2), nbRows), - QtCore.Qt.Key_PageUp: lambda current, nbRows: min(current - min(1, self.maxVisibleRows / 2), 0), - } - acceptKeys = { - QtCore.Qt.Key_Enter: 'select', - QtCore.Qt.Key_Return: 'select', - # space on a toolbar/category should toggle the entire category in the search results - QtCore.Qt.Key_Space: 'toggle', - } - cancelKeys = { - QtCore.Qt.Key_Escape: True, - } - - currentIndex = self.listView.currentIndex() - if key in listMovementKeys: - self.showList() - if self.listView.isEnabled(): - currentRow = currentIndex.row() - nbRows = self.listView.model().rowCount() - newRow = listMovementKeys[key](currentRow, nbRows) - index = self.listView.model().index(newRow, 0) - self.listView.setCurrentIndex(index) - elif key in acceptKeys: - self.showList() - if currentIndex.isValid(): - self.selectResult(acceptKeys[key], currentIndex, currentIndex.data()) - elif key in cancelKeys: - self.listView.hide() - self.clearFocus() - else: - self.showList() - super(SearchBox, self).keyPressEvent(qKeyEvent) - def showList(self): - def getScreenPosition(widget): - geo = widget.geometry() - print(geo) - parent = widget.parent() - parentPos = getScreenPosition(parent) if parent is not None else QtCore.QPoint(0,0) - return QtCore.QPoint(geo.x() + parentPos.x(), geo.y() + parentPos.y()) - pos = getScreenPosition(self) - siz = self.size() - screen = QtGui.QGuiApplication.screenAt(pos) - print(pos, siz, screen) - x = pos.x() - y = pos.y() + siz.height() - hint_w = self.listView.sizeHint().width() - # TODO: this can still bump into the bottom of the screen, in that case we should flip - w = max(siz.width(), hint_w) - h = 100 - if screen is not None: - scr = screen.geometry() - x = min(scr.x() + scr.width() - hint_w, x) - self.listView.setGeometry(x, y, w, h) - self.listView.show() - def selectResult(self): - self.listView.hide() - # TODO: allow other options, e.g. some items could act as combinators / cumulative filters - self.setText('') - self.filterList(self.text()) - def filterList(self, query): - print('TODO: do the actual filtering') - flt = QtCore.QSortFilterProxyModel() - flt.setSourceModel(self.model) - flt.setFilterCaseSensitivity(QtCore.Qt.CaseSensitivity.CaseInsensitive) - flt.setFilterWildcard(query) - self.listView.setModel(flt) - # TODO: try to find the already-highlighted item - nbRows = self.listView.model().rowCount() - if nbRows > 0: - index = self.listView.model().index(0, 0) - self.listView.setCurrentIndex(index) - #self.showList() - def showExtraInfo(selected, deselected): - print('show extra info...') - pass - mdl = QtCore.QStringListModel(['aaa', 'aab', 'aac', 'bxy', 'bac']) - sbx = SearchBox(mdl, 10, None) - sbx.show() \ No newline at end of file +import os +from PySide import QtGui +from PySide import QtCore + +globalIgnoreFocusOut = False + +genericToolIcon = QtGui.QIcon(QtGui.QIcon(os.path.dirname(__file__) + '/Tango-Tools-spanner-hammer.svg')) + +def easyToolTipWidget(html): + foo = QtGui.QTextEdit() + foo.setReadOnly(True) + foo.setAlignment(QtCore.Qt.AlignTop) + foo.setText(html) + return foo + +class SearchBox(QtGui.QLineEdit): + resultSelected = QtCore.Signal(int, int) + def __init__(self, getItemGroups, getToolTip, getItemDelegate, maxVisibleRows = 20, parent = None): + # Call parent cosntructor + super(SearchBox, self).__init__(parent) + # Save arguments + #self.model = model + self.getItemGroups = getItemGroups + self.getToolTip = getToolTip + self.itemGroups = None # Will be initialized by calling getItemGroups() the first time the search box gains focus, through focusInEvent and refreshItemGroups + self.maxVisibleRows = maxVisibleRows # TODO: use this to compute the correct height + # Create proxy model + self.proxyModel = QtCore.QIdentityProxyModel() + # Filtered model to which items are manually added. Store it as a property of the object instead of a local variable, to prevent grbage collection. + self.mdl = QtGui.QStandardItemModel() + #self.proxyModel.setModel(self.model) + # Create list view + self.listView = QtGui.QListView(self) + self.listView.setWindowFlags(QtGui.Qt.ToolTip) + self.listView.setWindowFlag(QtGui.Qt.FramelessWindowHint) + self.listView.setSelectionMode(QtGui.QAbstractItemView.SingleSelection) + self.listView.setModel(self.proxyModel) + self.listView.setItemDelegate(getItemDelegate()) # https://stackoverflow.com/a/65930408/324969 + # make the QListView non-editable + self.listView.setEditTriggers(QtGui.QAbstractItemView.NoEditTriggers) + # Create pane for showing extra info about the currently-selected tool + #self.extraInfo = QtGui.QLabel() + self.extraInfo = QtGui.QWidget() + self.extraInfo.setWindowFlags(QtGui.Qt.ToolTip) + self.extraInfo.setWindowFlag(QtGui.Qt.FramelessWindowHint) + self.extraInfo.setLayout(QtGui.QVBoxLayout()) + self.extraInfo.layout().setContentsMargins(0,0,0,0) + self.setExtraInfoIsActive = False + self.pendingExtraInfo = None + # Connect signals and slots + self.textChanged.connect(self.filterModel) + self.listView.clicked.connect(lambda x: self.selectResult('select', x)) + self.listView.selectionModel().selectionChanged.connect(self.onSelectionChanged) + # Thanks to https://saurabhg.com/programming/search-box-using-qlineedit/ for indicating a few useful options + ico = QtGui.QIcon(':/icons/help-browser.svg') + #ico = QtGui.QIcon(':/icons/WhatsThis.svg') + self.addAction(ico, QtGui.QLineEdit.LeadingPosition) + self.setClearButtonEnabled(True) + self.setPlaceholderText('Search tools, prefs & tree') + self.setFixedWidth(200) # needed to avoid a change of width when the clear button appears/disappears + # Initialize the model with the full list (assuming the text() is empty) + #self.filterModel(self.text()) # This is done by refreshItemGroups on focusInEvent, because the initial loading from cache can take time + def refreshItemGroups(self): + self.itemGroups = self.getItemGroups() + self.filterModel(self.text()) + def focusInEvent(self, qFocusEvent): + global globalIgnoreFocusOut + if not globalIgnoreFocusOut: + self.refreshItemGroups() + self.showList() + super(SearchBox, self).focusInEvent(qFocusEvent) + def focusOutEvent(self, qFocusEvent): + global globalIgnoreFocusOut + if not globalIgnoreFocusOut: + self.hideList() + super(SearchBox, self).focusOutEvent(qFocusEvent) + def keyPressEvent(self, qKeyEvent): + key = qKeyEvent.key() + listMovementKeys = { + QtCore.Qt.Key_Down: lambda current, nbRows: (current + 1) % nbRows, + QtCore.Qt.Key_Up: lambda current, nbRows: (current - 1) % nbRows, + QtCore.Qt.Key_PageDown: lambda current, nbRows: min(current + max(1, self.maxVisibleRows / 2), nbRows - 1), + QtCore.Qt.Key_PageUp: lambda current, nbRows: max(current - max(1, self.maxVisibleRows / 2), 0), + QtCore.Qt.Key_Home: lambda current, nbRows: 0, + QtCore.Qt.Key_End: lambda current, nbRows: nbRows - 1, + } + acceptKeys = { + QtCore.Qt.Key_Enter: 'select', + QtCore.Qt.Key_Return: 'select', + # space on a toolbar/category should toggle the entire category in the search results + QtCore.Qt.Key_Space: 'toggle', + } + cancelKeys = { + QtCore.Qt.Key_Escape: True, + } + + currentIndex = self.listView.currentIndex() + if key in listMovementKeys: + self.showList() + if self.listView.isEnabled(): + currentRow = currentIndex.row() + nbRows = self.listView.model().rowCount() + if nbRows > 0: + newRow = listMovementKeys[key](currentRow, nbRows) + index = self.listView.model().index(newRow, 0) + self.listView.setCurrentIndex(index) + elif key in acceptKeys: + self.showList() + if currentIndex.isValid(): + self.selectResult(acceptKeys[key], currentIndex) + elif key in cancelKeys: + self.hideList() + self.clearFocus() + else: + self.showList() + super(SearchBox, self).keyPressEvent(qKeyEvent) + def showList(self): + self.setFloatingWidgetsGeometry() + if not self.listView.isVisible(): + self.listView.show() + self.showExtraInfo() + def hideList(self): + self.listView.hide() + self.hideExtraInfo() + def hideExtraInfo(self): + self.extraInfo.hide() + def selectResult(self, mode, index): + groupIdx = int(index.model().itemData(index.siblingAtColumn(2))[0]) + self.hideList() + # TODO: allow other options, e.g. some items could act as combinators / cumulative filters + self.setText('') + self.filterModel(self.text()) + # TODO: emit index relative to the base model + self.resultSelected.emit(index, groupIdx) + def filterModel(self, userInput): + # TODO: this will cause a race condition if it is accessed while being modified + def matches(s): + return userInput.lower() in s.lower() + def filterGroup(group): + if matches(group['text']): + # If a group matches, include the entire subtree (might need to disable this if it causes too much noise) + return group + else: + subitems = filterGroups(group['subitems']) + if len(subitems) > 0 or matches(group['text']): + return { 'id': group['id'], 'text': group['text'], 'icon': group['icon'], 'action': group['action'], 'toolTip':group['toolTip'], 'subitems': subitems } + else: + return None + def filterGroups(groups): + groups = (filterGroup(group) for group in groups) + return [group for group in groups if group is not None] + self.mdl = QtGui.QStandardItemModel() + self.mdl.appendColumn([]) + def addGroups(filteredGroups, depth=0): + for group in filteredGroups: + # TODO: this is not very clean, we should memorize the index from the original itemgroups + self.mdl.appendRow([QtGui.QStandardItem(group['icon'] or genericToolIcon, group['text']), + QtGui.QStandardItem(str(depth)), + QtGui.QStandardItem(str(group['id']))]) + addGroups(group['subitems'], depth+1) + addGroups(filterGroups(self.itemGroups)) + self.proxyModel.setSourceModel(self.mdl) + # TODO: try to find the already-highlighted item + nbRows = self.listView.model().rowCount() + if nbRows > 0: + index = self.listView.model().index(0, 0) + self.listView.setCurrentIndex(index) + self.setExtraInfo(index) + else: + self.clearExtraInfo() + #self.showList() + def setFloatingWidgetsGeometry(self): + def getScreenPosition(widget): + geo = widget.geometry() + parent = widget.parent() + parentPos = getScreenPosition(parent) if parent is not None else QtCore.QPoint(0,0) + return QtCore.QPoint(geo.x() + parentPos.x(), geo.y() + parentPos.y()) + pos = getScreenPosition(self) + siz = self.size() + screen = QtGui.QGuiApplication.screenAt(pos) + x = pos.x() + y = pos.y() + siz.height() + hint_w = self.listView.sizeHint().width() + # TODO: this can still bump into the bottom of the screen, in that case we should flip + w = max(siz.width(), hint_w) + h = 200 # TODO: set height / size here according to desired number of items + extraw = w # choose a preferred width that doesn't change all the time, + # self.extraInfo.sizeHint().width() would change for every item. + extrax = x - extraw + if screen is not None: + scr = screen.geometry() + x = min(scr.x() + scr.width() - hint_w, x) + extraleftw = x - scr.x() + extrarightw = scr.x() + scr.width() - x + # flip the extraInfo if it doesn't fit on the screen + if extraleftw < extraw and extrarightw > extraleftw: + extrax = x + w + extraw = min(extrarightw, extraw) + else: + extrax = x - extraw + extraw = min(extraleftw, extraw) + self.listView.setGeometry(x, y, w, h) + self.extraInfo.setGeometry(extrax, y, extraw, h) + def onSelectionChanged(self, selected, deselected): + # The list has .setSelectionMode(QtGui.QAbstractItemView.SingleSelection), + # so there is always at most one index in selected.indexes() and at most one + # index in deselected.indexes() + selected = selected.indexes() + deselected = deselected.indexes() + if len(selected) > 0: + index = selected[0] + self.setExtraInfo(index) + # Poor attempt to circumvent a glitch where the extra info pane stays visible after pressing Return + if not self.listView.isHidden(): + self.showExtraInfo() + elif len(deselected) > 0: + self.hideExtraInfo() + def setExtraInfo(self, index): + # TODO: use an atomic swap or mutex if possible + if self.setExtraInfoIsActive: + self.pendingExtraInfo = index + #print("boom") + else: + self.setExtraInfoIsActive = True + #print("lock") + # setExtraInfo can be called multiple times while this function is running, + # so just before existing we check for the latest pending call and execute it, + # if during that second execution some other calls are made the latest of those will + # be queued by the code a few lines above this one, and the loop will continue processing + # until an iteration during which no further call was made. + while True: + groupId = str(index.model().itemData(index.siblingAtColumn(2))[0]) + # TODO: move this outside of this class, probably use a single metadata + # This is a hack to allow some widgets to set the parent and recompute their size + # during their construction. + parentIsSet = False + def setParent(toolTipWidget): + nonlocal parentIsSet + parentIsSet = True + w = self.extraInfo.layout().takeAt(0) + while w: + if hasattr(w.widget(), 'finalizer'): + # The 3D viewer segfaults very easily if it is used after being destroyed, and some Python/C++ interop seems to overzealously destroys some widgets, including this one, too soon? + # Ensuring that we properly detacth the 3D viewer widget before discarding its parent seems to avoid these crashes. + #print('FINALIZER') + w.widget().finalizer() + if w.widget() is not None: + w.widget().hide() # hide before detaching, or we have widgets floating as their own window that appear for a split second in some cases. + w.widget().setParent(None) + w = self.extraInfo.layout().takeAt(0) + self.extraInfo.layout().addWidget(toolTipWidget) + self.setFloatingWidgetsGeometry() + toolTipWidget = self.getToolTip(groupId, setParent) + if isinstance(toolTipWidget, str): + toolTipWidget = easyToolTipWidget(toolTipWidget) + if not parentIsSet: + setParent(toolTipWidget) + if self.pendingExtraInfo is not None: + index = self.pendingExtraInfo + self.pendingExtraInfo = None + else: + break + #print("unlock") + self.setExtraInfoIsActive = False + def clearExtraInfo(self): + # TODO: just clear the contents but keep the widget visible. + self.extraInfo.hide() + def showExtraInfo(self): + self.extraInfo.show() diff --git a/SearchResults.py b/SearchResults.py new file mode 100644 index 0000000..58252fa --- /dev/null +++ b/SearchResults.py @@ -0,0 +1,18 @@ +actionHandlers = { } +toolTipHandlers = { } +resultProvidersCached = { } +resultProvidersUncached = { } + +# name : string +# getItemGroupsCached: () -> [itemGroup] +# getItemGroupsUncached: () -> [itemGroup] +def registerResultProvider(name, getItemGroupsCached, getItemGroupsUncached): + resultProvidersCached[name] = getItemGroupsCached + resultProvidersUncached[name] = getItemGroupsUncached + +# name : str +# action : act -> None +# toolTip : groupId, setParent -> (str or QWidget) +def registerResultHandler(name, action, toolTip): + actionHandlers[name] = action + toolTipHandlers[name] = toolTip diff --git a/SearchTools.py b/SearchTools.py deleted file mode 100644 index 0949dd0..0000000 --- a/SearchTools.py +++ /dev/null @@ -1,812 +0,0 @@ -import os -import FreeCAD as App -import FreeCADGui -from PySide import QtGui -from PySide import QtCore - -""" -# Reload with: -import SearchTools; from importlib import reload; reload(SearchTools) - - -TODO for this project: -OK find a way to use the FreeCAD 3D viewer without segfaults or disappearing widgets -OK fix sync problem when moving too fast -OK split the list of tools vs. document objects -OK save to disk the list of tools -OK always display including when switching workbenches -OK slightly larger popup widget to avoid scrollbar for the extra info for document objects -OK turn this into a standalone mod -OK Optimize so that it's not so slow -OK speed up startup to show the box instantly and do the slow loading on first click. -OK One small bug: when the 3D view is initialized, it causes a loss of focus on the drop-down. We restore it, but the currently-selected index is left unchanged, so the down or up arrow has to be pressed twice. -* split into several files, try to keep the absolute minimum of code possible in the main file to speed up startup -OK segfault when reloading -* Disable the spacebar shortcut (can't type space in the search field…) -* Possibly disable the home and end, and use ctrl+home and ctrl+end instead? -""" - -################################"" - -class SafeViewer(QtGui.QWidget): - """FreeCAD uses a modified version of QuarterWidget, so the import pivy.quarter one will cause segfaults. - FreeCAD's FreeCADGui.createViewer() puts the viewer widget inside an MDI window, and detaching it without causing segfaults on exit is tricky. - This class contains some kludges to extract the viewer as a standalone widget and destroy it safely.""" - def __init__(self, parent = None): - super(SafeViewer, self).__init__() - self.viewer = FreeCADGui.createViewer() - self.graphicsView = self.viewer.graphicsView() - self.oldGraphicsViewParent = self.graphicsView.parent() - self.oldGraphicsViewParentParent = self.oldGraphicsViewParent.parent() - self.oldGraphicsViewParentParentParent = self.oldGraphicsViewParentParent.parent() - - # Avoid segfault but still hide the undesired window by moving it to a new hidden MDI area. - self.hiddenQMDIArea = QtGui.QMdiArea() - self.hiddenQMDIArea.addSubWindow(self.oldGraphicsViewParentParentParent) - - self.private_widget = self.oldGraphicsViewParent - self.private_widget.setParent(parent) - - self.setLayout(QtGui.QVBoxLayout()) - self.layout().addWidget(self.private_widget) - self.layout().setContentsMargins(0,0,0,0) - - def fin(slf): - slf.finalizer() - - import weakref - weakref.finalize(self, fin, self) - - self.destroyed.connect(self.finalizer) - - def finalizer(self): - # Cleanup in an order that doesn't cause a segfault: - self.private_widget.setParent(self.oldGraphicsViewParentParent) - self.oldGraphicsViewParentParentParent.close() - self.oldGraphicsViewParentParentParent = None - self.oldGraphicsViewParentParent = None - self.oldGraphicsViewParent = None - self.graphicsView = None - self.viewer = None - #self.parent = None - self.hiddenQMDIArea = None - -""" -# Example use: -from PySide import QtGui -import pivy -def mk(v): - w = QtGui.QMainWindow() - oldFocus = QtGui.QApplication.focusWidget() - sv.widget.setParent(w) - oldFocus.setFocus() - w.show() - col = pivy.coin.SoBaseColor() - col.rgb = (1, 0, 0) - trans = pivy.coin.SoTranslation() - trans.translation.setValue([0, 0, 0]) - cub = pivy.coin.SoCube() - myCustomNode = pivy.coin.SoSeparator() - myCustomNode.addChild(col) - myCustomNode.addChild(trans) - myCustomNode.addChild(cub) - sv.viewer.getViewer().setSceneGraph(myCustomNode) - sv.viewer.fitAll() - return w -sv = SafeViewer() -ww=mk(sv) -""" - -genericToolIcon = QtGui.QIcon(QtGui.QIcon(os.path.dirname(__file__) + '/Tango-Tools-spanner-hammer.svg')) - -def iconToBase64(icon, sz = QtCore.QSize(64,64), mode = QtGui.QIcon.Mode.Normal, state = QtGui.QIcon.State.On): - buf = QtCore.QBuffer() - buf.open(QtCore.QIODevice.WriteOnly) - icon.pixmap(sz, mode, state).save(buf, 'PNG') - return QtCore.QTextCodec.codecForName('UTF-8').toUnicode(buf.data().toBase64()) -def iconToHTML(icon, sz = 12, mode = QtGui.QIcon.Mode.Normal, state = QtGui.QIcon.State.On): - return '' + nfo['toolTip']['label'] + '
App.getDocument(' + repr(str(nfo['toolTip']['docName'])) + ').getObject(' + repr(str(nfo['toolTip']['name'])) + ')
Load all workbenches to refresh this list of tools. This may take a minute, depending on the number of installed workbenches.
') -def toolbarToolTip(nfo, setParent): - return easyToolTipWidget('Display toolbar ' + nfo['toolTip'] + '
This toolbar appears in the following workbenches:
' + nfo['toolTip'] + '
') -def documentObjectToolTip(nfo, setParent): - return DocumentObjectToolTipWidget(nfo, setParent) -def documentToolTip(nfo, setParent): - return easyToolTipWidget('' + nfo['toolTip']['label'] + '
App.getDocument(' + repr(str(nfo['toolTip']['name'])) + ')