PySide(PyQT)重新定义contextMenuEvent()实现鼠标右键弹出菜单

在 PySide中,contextMenuEvent()QWidget 类(以及继承自它的所有子类)的一个事件处理方法,主要用于处理上下文菜单事件,也就是当用户在控件上右键点击时触发的事件。

• 通过重新定义contextMenuEvent()来实现自定义的鼠标右键菜单

python 复制代码
from PySide6.QtGui import QAction
from PySide6.QtWidgets import QApplication, QWidget, QMenu
from PySide6.QtCore import Qt


class MyWidget(QWidget):
    def __init__(self):
        super().__init__()
        # 创建一个菜单
        self.menu = QMenu(self)

        # 创建菜单项
        action1 = QAction("菜单项1", self)
        action2 = QAction("菜单项2", self)

        # 将菜单项添加到菜单中
        self.menu.addAction(action1)
        self.menu.addAction(action2)

        # 连接信号和槽,定义点击菜单项后的行为
        action1.triggered.connect(self.on_action1_triggered)
        action2.triggered.connect(self.on_action2_triggered)

    def contextMenuEvent(self, event):

        # 在鼠标点击的位置显示菜单
        self.menu.exec(event.globalPos())

    def on_action1_triggered(self):
        print("菜单项1被触发")

    def on_action2_triggered(self):
        print("菜单项2被触发")


# 应用程序执行部分
if __name__ == "__main__":
    app = QApplication([])
    widget = MyWidget()
    widget.show()
    app.exec()

• 将其定义在子部件内:

python 复制代码
from PySide6.QtGui import QAction
from PySide6.QtWidgets import QApplication, QWidget, QMenu, QPushButton


class MyWidget(QWidget):
    def __init__(self):
        super().__init__()
        self.setupUi()

    def setupUi(self):
        self.button = MyButton(self)

class MyButton(QPushButton):
    def __init__(self, parent=None):
        super().__init__(parent)
        self.setText("右键点击我")
        # 创建一个菜单
        self.menu = QMenu(self)

        # 创建菜单项
        action1 = QAction("菜单项1", self)
        action2 = QAction("菜单项2", self)

        # 将菜单项添加到菜单中
        self.menu.addAction(action1)
        self.menu.addAction(action2)

        # 连接信号和槽,定义点击菜单项后的行为
        action1.triggered.connect(self.on_action1_triggered)
        action2.triggered.connect(self.on_action2_triggered)

    def contextMenuEvent(self, event):

        # 在鼠标点击的位置显示菜单
        self.menu.exec(event.globalPos())

    def on_action1_triggered(self):
        print("菜单项1被触发")

    def on_action2_triggered(self):
        print("菜单项2被触发")


# 应用程序执行部分
if __name__ == "__main__":
    app = QApplication([])
    widget = MyWidget()
    widget.show()
    app.exec()

上述代码,只有在按钮上有鼠标右键菜单。

获取系统默认的右键菜单并添加新的菜单项

python 复制代码
import sys
from PySide6.QtGui import QAction
from PySide6.QtWidgets import QApplication, QLineEdit, QMenu

class MyLineEdit(QLineEdit):
    def __init__(self, parent=None):
        super().__init__(parent)
        self.setText("右键点击我")

        self.menu = QMenu(self)   # 创建一个菜单
        self.addMenus()         # 添加菜单的菜单项

    def addMenus(self):
        # 获取系统默认的上下文菜单动作
        default_actions = self.createStandardContextMenu().actions()
        # 将默认动作添加到新菜单中
        for action in default_actions:
            print(action.text())
            self.menu.addAction(action)
        # 将新的菜单项添加到新菜单中
        self.action1 = QAction("菜单项1", self)
        self.action2 = QAction("菜单项2", self)
        self.menu.addSeparator()  # 创建分割线
        # 添加自定义菜单项
        self.menu.addAction(self.action1)
        self.menu.addAction(self.action2)
        # 连接信号和槽函数
        self.action1.triggered.connect(self.on_action1_triggered)
        self.action2.triggered.connect(self.on_action2_triggered)


    # 重写上下文菜单事件
    def contextMenuEvent(self, event):
        # 显示菜单
        self.menu.exec(event.globalPos())

    def on_action1_triggered(self):
        print("菜单项1被点击")

    def on_action2_triggered(self):
        print("菜单项2被点击")


if __name__ == "__main__":
    app = QApplication(sys.argv)
    line_edit = MyLineEdit()
    line_edit.show()
    sys.exit(app.exec())

上面的代码,读取了系统内置的默认右键菜单,并增加了两个自定义的菜单项,构成了一个新的菜单。重写了contextMenuEvent(),使其显示新的菜单。

为新菜单设置快捷方式

python 复制代码
import sys
from PySide6.QtGui import QAction
from PySide6.QtWidgets import QApplication, QLineEdit, QMenu

class MyLineEdit(QLineEdit):
    def __init__(self, parent=None):
        super().__init__(parent)
        self.setText("右键点击我")

        self.menu = QMenu(self)   # 创建一个菜单
        self.addMenus()         # 添加菜单的菜单项


    def addMenus(self):
        # 获取系统默认的上下文菜单动作
        default_actions = self.createStandardContextMenu().actions()
        # 将默认动作添加到新菜单中
        for action in default_actions:
            self.menu.addAction(action)
        # 将新的菜单项添加到新菜单中
        self.action1 = QAction("菜单项1", self)   # 创建菜单项
        self.action1.setShortcut("Ctrl+B")       # 设置快捷键
        self.action2 = QAction("菜单项2", self)
        self.menu.addSeparator()  # 创建分割线
        # 添加自定义菜单项
        self.addAction(self.action1)             # 将菜单项添加到部件的主体中,用以生效快捷方式
        self.menu.addAction(self.action1)  # 将菜单项添加到菜单中
        self.menu.addAction(self.action2)
        # 连接信号和槽函数
        self.action1.triggered.connect(self.on_action1_triggered)
        self.action2.triggered.connect(self.on_action2_triggered)


    # 重写上下文菜单事件
    def contextMenuEvent(self, event):
        # 显示菜单
        self.menu.exec(event.globalPos())

    def on_action1_triggered(self):
        print("菜单项1被点击")

    def on_action2_triggered(self):
        print("菜单项2被点击")


if __name__ == "__main__":
    app = QApplication(sys.argv)
    line_edit = MyLineEdit()
    line_edit.show()
    sys.exit(app.exec())
复制代码
    这里注意,只有执行了:self.addAction(self.action1),将菜单项添加到部件的主体中,才能生效快捷方式。而self.menu.addAction(self.action1)只是将菜单项添加到菜单中,并未在部件中注册菜单项。

除了重新定义contextMenuEvent(),还有另一种更为灵活的方法:

• 使用策略设置setContextMenuPolicy()的方法来定义菜单:

PyQT(PySide)的上下文菜单策略设置setContextMenuPolicy()-CSDN博客

相关推荐
witAI5 分钟前
手机生成剧本杀软件2025推荐,创新剧情设计工具助力创作
人工智能·python
zzginfo11 分钟前
JavaScript 解构赋值
开发语言·javascript·ecmascript
2501_9216494915 分钟前
Java 接入外汇数据 API 完整教程:实时报价、历史 K 线与 WebSocket 推送
java·开发语言·websocket·金融
℡終嚸♂68017 分钟前
Java 反序列化漏洞详解
java·开发语言
故事和你9122 分钟前
蓝桥杯-2025年C++B组国赛
开发语言·软件测试·数据结构·c++·算法·职场和发展·蓝桥杯
派大星~课堂29 分钟前
【力扣-138. 随机链表的复制 ✨】Python笔记
python·leetcode·链表
王忘杰31 分钟前
0基础CUDA炼丹、增加断点保存,从零开始训练自己的AI大模型 87owo/EasyGPT Python CUDA
开发语言·人工智能·python
数据知道33 分钟前
claw-code 源码详细分析:`reference_data` JSON 快照——大型移植里「对照底稿」该怎么治理与演进?
linux·python·ubuntu·json·claude code
好家伙VCC34 分钟前
**发散创新:基于以太坊侧链的高性能去中心化应用部署实战**在区块链生态中,*
java·python·去中心化·区块链
Lzh编程小栈38 分钟前
数据结构与算法之队列深度解析:循环队列+C 语言硬核实现 + 面试考点全梳理
c语言·开发语言·汇编·数据结构·后端·算法·面试