如何说服鼠标右键单击弹出菜单在wxPython中显示靠近鼠标光标位置?

问题描述

我有一个使用 SplitterWindow 的 wx 框架(该框架还有主菜单、工具栏、状态栏——但这与此处无关)。一切都按预期工作,除了按钮上的鼠标右键单击弹出菜单,因为弹出菜单显示在屏幕上明显随机的位置 - 以及将框架移动到第二个屏幕(监视器)时的“负”随机位置。 “显然”是指弹出菜单的位置似乎与实际按钮(或框架)位置有些相关,但乘以某些因素 - 在主屏幕上为正,在第二个屏幕上为负。

我仅在 Windows 10 64 位 / Python 3.9.0 64 位 / wx '4.1.1 msw (phoenix) wxWidgets 3.1.5' 上运行代码。框架代码的第一行是通过 wxGlade 生成的,所以这可能与最初生成框架代码的特定方式有关。

我创建了一个精简的测试代码,如下所示,根据鼠标右键单击弹出菜单模拟真实代码的确切情况。在此测试代码中,我将按钮放置在第二个窗格中,但它在任何窗格中的行为都相同。

我在其他简单的 wx 示例代码上尝试了相同的弹出菜单代码,但没有使用 SplitterWindow,并且弹出行为正常。我应该在下面的测试代码中更改或改进什么?

#!/usr/bin/env python3
# -*- coding: UTF-8 -*-

import wx

class MyFrame(wx.Frame):
    def __init__(self,*args,**kwds):
        kwds["style"] = kwds.get("style",0) | wx.DEFAULT_FRAME_STYLE
        wx.Frame.__init__(self,**kwds)
        self.SetSize((400,300))
        self.SetTitle("test frame")

        sizer_0 = wx.BoxSizer(wx.HORIZONTAL)
        self.window_1 = wx.SplitterWindow(self,wx.ID_ANY)
        self.window_1.SetMinimumPanesize(20)
        sizer_0.Add(self.window_1,1,wx.EXPAND,0)

        self.pane_1 = wx.Panel(self.window_1,wx.ID_ANY)
        sizer_1 = wx.BoxSizer(wx.HORIZONTAL)
        self.pane_2 = wx.Panel(self.window_1,wx.ID_ANY)
        sizer_2 = wx.BoxSizer(wx.HORIZONTAL)

        self.button = wx.Button(self.pane_2,wx.ID_ANY,"test button")
        sizer_2.Add(self.button,0)

        self.pane_1.SetSizer(sizer_1)
        self.pane_2.SetSizer(sizer_2)
        self.window_1.SplitVertically(self.pane_1,self.pane_2)
        self.SetSizer(sizer_0)
        self.Layout()

        self.Bind(wx.EVT_CONTEXT_MENU,self.OnButtonContextMenu,self.button)

    def OnButtonContextMenu(self,event):
        self.PopupMenu(ButtonContext(self),event.GetPosition())

##

class ButtonContext(wx.Menu):
    def __init__(self,parent):
        super(ButtonContext,self).__init__()
        self.parent = parent
 
        button_popup = wx.MenuItem(self,'test popup')
        self.Append(button_popup)
        self.Bind(wx.EVT_MENU,self.button_action,button_popup)
 
    def button_action(self,event):
        event.Skip()

##

class MyApp(wx.App):
    def OnInit(self):
        self.frame = MyFrame(None,"")
        self.SetTopWindow(self.frame)
        self.frame.Show()
        return True

##

if __name__ == "__main__":
    app = MyApp(0)
    app.MainLoop()

解决方法

这有点令人费解,但我认为您通过将 InvokingWindow 传递给 wx.Menu 来覆盖 event.GetPosition()class ButtonContext 的位置。

简而言之,如果您删除该参数 event.GetPosition() 并仅使用 self.PopupMenu(ButtonContext(self)) 调用它,它将默认为父窗口,即按钮本身。
结果是它总是聚焦在你刚刚右击的按钮上。

enter image description here