QMenu断开的信号或事件

时间:2019-04-09 18:37:32

标签: python events pyqt signals-slots qmenu

是否有信号/事件可用于QMenu拆卸?

我有一个QMenu子类,其中有.setTearOffEnabled(True),但是如果用户单击撕开的“条”,我想将此撕开部分设置为始终位于顶部。

我无法使用QtCore.Qt.WindowStaysOnTopHint,因为这将导致我的菜单已经处于可拆卸状态。

例如:如果我的主要工具区域大于撕开的区域,然后单击我的主要工具,则撕开窗口将在其后面。

2 个答案:

答案 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)