是否有信号/事件可用于QMenu拆卸?
我有一个QMenu
子类,其中有.setTearOffEnabled(True)
,但是如果用户单击撕开的“条”,我想将此撕开部分设置为始终位于顶部。
我无法使用QtCore.Qt.WindowStaysOnTopHint
,因为这将导致我的菜单已经处于可拆卸状态。
例如:如果我的主要工具区域大于撕开的区域,然后单击我的主要工具,则撕开窗口将在其后面。
答案 0 :(得分:1)
在下面的代码中,当按下 tear off (虚线)时,发出咔嗒声:
import sys
from PyQt5 import QtCore, QtWidgets
class Menu(QtWidgets.QMenu):
clicked = QtCore.pyqtSignal()
def mouseReleaseEvent(self, event):
if self.isTearOffEnabled():
tearRect = QtCore.QRect(
0,
0,
self.width(),
self.style().pixelMetric(
QtWidgets.QStyle.PM_MenuTearoffHeight, None, self
),
)
if tearRect.contains(event.pos()):
self.clicked.emit()
QtCore.QTimer.singleShot(0, self.after_clicked)
super(Menu, self).mouseReleaseEvent(event)
@QtCore.pyqtSlot()
def after_clicked(self):
tornPopup = None
for tl in QtWidgets.QApplication.topLevelWidgets():
if tl.metaObject().className() == "QTornOffMenu":
tornPopup = tl
break
if tornPopup is not None:
print("This is the tornPopup: ", tornPopup)
tornPopup.setWindowFlag(QtCore.Qt.WindowStaysOnTopHint)
if __name__ == "__main__":
app = QtWidgets.QApplication(sys.argv)
w = QtWidgets.QMainWindow(parent=None)
menu = Menu("Menu", tearOffEnabled=True)
menu.clicked.connect(lambda: print("clicked"))
w.menuBar().addMenu(menu)
for i in range(5):
action = QtWidgets.QAction("action{}".format(i), w)
menu.addAction(action)
w.show()
sys.exit(app.exec_())
答案 1 :(得分:0)
当菜单被撕下时,它会被隐藏,并且Qt会用从QMenu
的内部子类创建的副本替换它。因此,要在撕下的菜单上设置WindowStaysOnTopHint
,首先需要找到一种方法来引用它。一种方法是在应用程序对象上设置event-filter并注意正确类型的child events:
class MenuWatcher(QtCore.QObject):
def __init__(self, parent=None):
super().__init__(parent)
QtWidgets.qApp.installEventFilter(self)
def eventFilter(self, source, event):
if (event.type() == QtCore.QEvent.ChildAdded and
event.child().metaObject().className() == 'QTornOffMenu'):
event.child().setWindowFlag(QtCore.Qt.WindowStaysOnTopHint)
return super().eventFilter(source, event)
此类将在应用程序中创建的所有撕掉菜单上操作。
但是,如果事件过滤是由源菜单类完成的,则可以通过比较菜单项来标识其自身的撕下菜单:
class Menu(QtWidgets.QMenu):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs)
self.setTearOffEnabled(True)
QtWidgets.qApp.installEventFilter(self)
def eventFilter(self, source, event):
if event.type() == QtCore.QEvent.ChildAdded:
child = event.child()
if (child.metaObject().className() == 'QTornOffMenu' and
all(a is b for a, b in zip(child.actions(), self.actions()))):
child.setWindowFlag(QtCore.Qt.WindowStaysOnTopHint)
return super().eventFilter(source, event)