我正在将我的(嵌套菜单(QMenu 对象"转换"为 QAction(使用 setMenu
(,以便我可以切换它们的复选框,但复选框无法切换,它只是保持选中状态。
import functools
import sys
from PyQt4 import QtGui, QtCore
class QSubAction(QtGui.QAction):
def __init__(self, text="", parent=None):
super(QSubAction, self).__init__(text, parent)
self.setCheckable(True)
self.setChecked(True)
class QAddAction(QtGui.QAction):
def __init__(self, icon=None, text="Add Item", parent=None):
if icon:
super(QAddAction, self).__init__(icon, text, parent)
else:
super(QAddAction, self).__init__(text, parent)
class QCustomMenu(QtGui.QMenu):
"""Customized QMenu."""
def __init__(self, title, parent=None):
super(QCustomMenu, self).__init__(title=str(title), parent=parent)
self.setup_menu()
def setup_menu(self):
self.setContextMenuPolicy(QtCore.Qt.DefaultContextMenu)
def contextMenuEvent(self, event):
no_right_click = [QAddAction]
if any([isinstance(self.actionAt(event.pos()), instance) for instance in no_right_click]):
return
pos = event.pos()
def addAction(self, action):
super(QCustomMenu, self).addAction(action)
class Example(QtGui.QMainWindow):
def __init__(self, parent=None):
super(Example, self).__init__(parent)
self.initUI()
def initUI(self):
self.setGeometry(300, 300, 300, 200)
self.setWindowTitle('Context menu')
self.qmenu = QCustomMenu(title='', parent=self)
add_item_action = QtGui.QAction('Add Main item', self,
triggered=self.add_new_item)
self.qmenu.addAction(add_item_action)
def contextMenuEvent(self, event):
action = self.qmenu.exec_(self.mapToGlobal(event.pos()))
def add_new_item(self):
main_menu_name, ok = QtGui.QInputDialog.getText(
self,
'Main Menu',
'Name of new Menu Item:'
)
if ok:
self._addMenuItemTest(main_menu_name)
def _addMenuItemTest(self, main_menu_name):
icon_path = '/user_data/add.png'
base_qmenu = QCustomMenu(title=main_menu_name, parent=self)
base_qmenu.setTearOffEnabled(True)
add_item_action = QAddAction(None, 'Add Sub Item', base_qmenu)
slot = functools.partial(self.add_sub_item, base_qmenu)
add_item_action.triggered.connect(slot)
base_qmenu.addAction(add_item_action)
# self.qmenu.addMenu(base_qmenu)
test_action = QtGui.QAction(main_menu_name, self)
test_action.setMenu(base_qmenu)
test_action.setCheckable(True)
test_action.setChecked(True)
self.qmenu.addAction(test_action)
def add_sub_item(self, base_menu):
sub_menu_name, ok = QtGui.QInputDialog.getText(
self,
'Sub Menu',
'Name of new Sub Item:'
)
if ok:
action = QSubAction(sub_menu_name, self)
slot = functools.partial(
self._callActionItem,
str(base_menu.title()),
str(sub_menu_name)
)
action.setCheckable(True)
action.setChecked(True)
action.toggled.connect(slot)
base_menu.addAction(action)
def _callActionItem(self, title, menu):
pass
if __name__ == "__main__":
app = QtGui.QApplication(sys.argv)
window = Example()
window.show()
sys.exit(app.exec_())
我通常使用PySide
。我试图通过使用PySide
来解决这个问题。
结果,请查看此代码并执行它。
import functools
import sys
from PyQt4 import QtGui, QtCore
class QSubAction(QtGui.QAction):
def __init__(self, text="", parent=None):
super(QSubAction, self).__init__(text, parent)
self.setCheckable(True)
self.setChecked(True)
class QAddAction(QtGui.QAction):
def __init__(self, icon=None, text="Add Item", parent=None):
if icon:
super(QAddAction, self).__init__(icon, text, parent)
else:
super(QAddAction, self).__init__(text, parent)
class QCustomMenu(QtGui.QMenu):
"""Customized QMenu."""
def __init__(self, title, parent=None):
super(QCustomMenu, self).__init__(title=str(title), parent=parent)
self.setup_menu()
def mousePressEvent(self,event):
action = self.activeAction()
if isinstance(action,QtGui.QAction):
action.trigger()
return QtGui.QMenu.mousePressEvent(self,event)
def setup_menu(self):
self.setContextMenuPolicy(QtCore.Qt.DefaultContextMenu)
def contextMenuEvent(self, event):
no_right_click = [QAddAction]
if any([isinstance(self.actionAt(event.pos()), instance) for instance in no_right_click]):
return
pos = event.pos()
def addAction(self, action):
super(QCustomMenu, self).addAction(action)
class Example(QtGui.QMainWindow):
def __init__(self, parent=None):
super(Example, self).__init__(parent)
self.initUI()
def initUI(self):
self.setGeometry(300, 300, 300, 200)
self.setWindowTitle('Context menu')
self.qmenu = QCustomMenu(title='', parent=self)
add_item_action = QtGui.QAction('Add Main item', self,
triggered=self.add_new_item)
self.qmenu.addAction(add_item_action)
def contextMenuEvent(self, event):
action = self.qmenu.exec_(self.mapToGlobal(event.pos()))
def add_new_item(self):
main_menu_name, ok = QtGui.QInputDialog.getText(
self,
'Main Menu',
'Name of new Menu Item:'
)
if ok:
self._addMenuItemTest(main_menu_name)
def _addMenuItemTest(self, main_menu_name):
icon_path = '/user_data/add.png'
base_qmenu = QCustomMenu(title=main_menu_name, parent=self)
base_qmenu.setTearOffEnabled(True)
add_item_action = QAddAction(None, 'Add Sub Item', base_qmenu)
slot = functools.partial(self.add_sub_item, base_qmenu)
add_item_action.triggered.connect(slot)
base_qmenu.addAction(add_item_action)
# self.qmenu.addMenu(base_qmenu)
test_action = QtGui.QAction(main_menu_name, self)
test_action.setMenu(base_qmenu)
test_action.setCheckable(True)
test_action.setChecked(True)
self.connect(test_action,QtCore.SIGNAL("triggered(bool)"),self.unsetCheck_action)
self.qmenu.addAction(test_action)
def unsetCheck_action(self,checked):
sender_obj = self.sender()
if isinstance(sender_obj,QtGui.QAction):
sender_obj.setChecked(checked)
def add_sub_item(self, base_menu):
sub_menu_name, ok = QtGui.QInputDialog.getText(
self,
'Sub Menu',
'Name of new Sub Item:'
)
if ok:
action = QSubAction(sub_menu_name, self)
slot = functools.partial(
self._callActionItem,
str(base_menu.title()),
str(sub_menu_name)
)
action.setCheckable(True)
action.setChecked(True)
action.toggled.connect(slot)
base_menu.addAction(action)
def _callActionItem(self, title, menu):
pass
if __name__ == "__main__":
app = QtGui.QApplication(sys.argv)
window = Example()
window.show()
sys.exit(app.exec_())
变化点。
1.自定义菜单具有鼠标按下事件。
def mousePressEvent(self,event):
action = self.activeAction()
if isinstance(action,QtGui.QAction):
action.trigger()
return QtGui.QMenu.mousePressEvent(self,event)
如果单击菜单上的操作,则会获得该操作。你打电话给action.trigger()
您可以连接action.triggered
信号。是的,"触发"是在"触发"之后。
2.连接触发器。
self.connect(test_action,QtCore.SIGNAL("triggered(bool)"),self.unsetCheck_action)
def unsetCheck_action(self,checked):
sender_obj = self.sender()
if isinstance(sender_obj,QtGui.QAction):
sender_obj.setChecked(checked)
如您所见,如果触发了测试动作,则会调用触发的信号。而且老虎机是unsetCheck_action
方法。
对于改变动作chack,你只能操纵似乎触发,而不需要在真和假之间切换。
法尔德在皮赛德
在 PyQt 中,最好通过 test_action.triggered[bool].connect()
但我无法通过PySide
做到这一点.
我的机器里没有PyQt4
。 如果您有一些问题,请随时向我询问更多信息。