wx Python小部件是否对整个窗口(全局)中的鼠标单击做出反应?
使用下面的代码段,生成如下内容: 。。。我想让蓝色(子类面板)小部件变成红色,每次用户用鼠标左键单击应用程序框架窗口中的任何位置。就目前的情况而言,它只在小部件本身被点击时才会做出反应——但它不会对点击按钮或框架中的其他地方做出反应 所以我的问题是:wx Python小部件是否对整个窗口(全局)中的鼠标单击做出反应?,python,wxpython,wxwidgets,Python,Wxpython,Wxwidgets,使用下面的代码段,生成如下内容: 。。。我想让蓝色(子类面板)小部件变成红色,每次用户用鼠标左键单击应用程序框架窗口中的任何位置。就目前的情况而言,它只在小部件本身被点击时才会做出反应——但它不会对点击按钮或框架中的其他地方做出反应 所以我的问题是: 在wxPython中,这个小部件是否可以监听“全局”点击 如果是-在下面的示例中如何执行 (仅仅更改父项,例如,self.parent.Bind(wx.EVT\u MOUSE\u EVENTS,self.OnMouseEvents)根本不起作
- 在wxPython中,这个小部件是否可以监听“全局”点击
- 如果是-在下面的示例中如何执行李>
self.parent.Bind(wx.EVT\u MOUSE\u EVENTS,self.OnMouseEvents)
根本不起作用;我也找到了,但我根本找不到正确的语法来设置它,使其起作用)
守则:
#!/usr/bin/env python
# -*- coding: utf-8 -*-
# generated by wxGlade 0.6.3 on Mon Jan 12 21:20:22 2015
import wx
# begin wxGlade: extracode
# end wxGlade
class MyPanel(wx.Panel):
def __init__(self, parent, id=wx.ID_ANY, pos=wx.DefaultPosition, size=wx.DefaultSize, style=0, name="MyPanel"):
super(MyPanel, self).__init__(parent, id, pos, size, style, name)
self.parent = parent
self.lmb = False
self.Bind(wx.EVT_MOUSE_EVENTS, self.OnMouseEvents) # only reacts on its own clicks, not globally!
self.Bind(wx.EVT_PAINT, self.OnPaint)
def OnMouseEvents(self, event):
print("OnMouseEvents")
self.lmb = event.LeftDown() or event.LeftIsDown()
self.Refresh(False)
def OnPaint(self, event):
w, h = self.GetSize()
if self.lmb: mybrush = wx.Brush(wx.NamedColour("red"))
else: mybrush = wx.Brush(wx.NamedColour("blue"))
dc = wx.PaintDC(self)
dc.SetPen(wx.TRANSPARENT_PEN)
dc.SetBrush(mybrush)
dc.DrawRectangle(0, 0, w, h)
class MyFrame(wx.Frame):
def __init__(self, *args, **kwds):
# begin wxGlade: MyFrame.__init__
kwds["style"] = wx.DEFAULT_FRAME_STYLE
wx.Frame.__init__(self, *args, **kwds)
self.button_1 = wx.Button(self, -1, "button_1")
self.panel_1 = MyPanel(self, -1)
self.__set_properties()
self.__do_layout()
# end wxGlade
def __set_properties(self):
# begin wxGlade: MyFrame.__set_properties
self.SetTitle("frame_1")
self.SetSize((200, 200))
# end wxGlade
def __do_layout(self):
# begin wxGlade: MyFrame.__do_layout
sizer_1 = wx.BoxSizer(wx.VERTICAL)
sizer_1.Add(self.button_1, 0, 0, 0)
sizer_1.Add(self.panel_1, 1, wx.EXPAND, 0)
self.SetSizer(sizer_1)
self.Layout()
# end wxGlade
# end of class MyFrame
if __name__ == "__main__":
app = wx.PySimpleApp(0)
wx.InitAllImageHandlers()
frame_1 = MyFrame(None, -1, "")
app.SetTopWindow(frame_1)
frame_1.Show()
app.MainLoop()
#!/usr/bin/env python
# -*- coding: utf-8 -*-
# generated by wxGlade 0.6.3 on Mon Jan 12 21:20:22 2015
import wx
# begin wxGlade: extracode
# end wxGlade
class MyPanel(wx.Panel):
def __init__(self, parent, id=wx.ID_ANY, pos=wx.DefaultPosition, size=wx.DefaultSize, style=0, name="MyPanel"):
super(MyPanel, self).__init__(parent, id, pos, size, style, name)
self.lmb = False
#self.Bind(wx.EVT_MOUSE_EVENTS, self.OnMouseEvents) # only reacts on its own clicks, not globally! # use RecursiveSetListeners now; should be called from parent class after all init layout is done
self.Bind(wx.EVT_PAINT, self.OnPaint)
def OnMouseEvents(self, event):
print("OnMouseEvents")
self.lmb = event.LeftDown() or event.LeftIsDown()
self.Refresh(False)
event.Skip() # propagate, in case button is clicked
def OnPaint(self, event):
w, h = self.GetSize()
if self.lmb: mybrush = wx.Brush(wx.NamedColour("red"))
else: mybrush = wx.Brush(wx.NamedColour("blue"))
dc = wx.PaintDC(self)
dc.SetPen(wx.TRANSPARENT_PEN)
dc.SetBrush(mybrush)
dc.DrawRectangle(0, 0, w, h)
def RecursiveSetListeners(self, inwidget):
print("setting listeners: " + inwidget.GetName())
inwidget.Connect(wx.ID_ANY, -1, wx.wxEVT_LEFT_DOWN, self.OnMouseEvents)
inwidget.Connect(wx.ID_ANY, -1, wx.wxEVT_LEFT_UP, self.OnMouseEvents)
for child in inwidget.GetChildren():
self.RecursiveSetListeners(child)
class MyFrame(wx.Frame):
def __init__(self, *args, **kwds):
# begin wxGlade: MyFrame.__init__
kwds["style"] = wx.DEFAULT_FRAME_STYLE
wx.Frame.__init__(self, *args, **kwds)
self.button_1 = wx.Button(self, -1, "button_1", name="button_1")
self.panel_1 = MyPanel(self, -1, name="panel_1")
self.button_2 = wx.Button(self.panel_1, -1, "button_2", name="button_1")
#self.Connect(wx.ID_ANY, -1, wx.EVT_MOUSE_EVENTS, self.panel_1.OnMouseEvents) # 'wx.wxEVT_MOUSE_EVENTS: AttributeError: 'module' has no attribute 'wxEVT_MOUSE_EVENTS';; wx.EVT_MOUSE_EVENTS: TypeError: in method 'EvtHandler_Connect', expected argument 4 of type 'int'
# these work:
#self.Connect(wx.ID_ANY, -1, wx.wxEVT_LEFT_DOWN, self.panel_1.OnMouseEvents)
#self.Connect(wx.ID_ANY, -1, wx.wxEVT_LEFT_UP, self.panel_1.OnMouseEvents)
#self.button_1.Connect(wx.ID_ANY, -1, wx.wxEVT_LEFT_DOWN, self.panel_1.OnMouseEvents)
#self.button_1.Connect(wx.ID_ANY, -1, wx.wxEVT_LEFT_UP, self.panel_1.OnMouseEvents)
self.__set_properties()
self.__do_layout()
print(self.GetChildren())
self.panel_1.RecursiveSetListeners(self)
# end wxGlade
def __set_properties(self):
# begin wxGlade: MyFrame.__set_properties
self.SetTitle("frame_1")
self.SetSize((200, 200))
# end wxGlade
def __do_layout(self):
# begin wxGlade: MyFrame.__do_layout
sizer_2 = wx.BoxSizer(wx.HORIZONTAL)
sizer_2.Add(self.button_2, 0,0,0)
self.panel_1.SetSizer(sizer_2)
sizer_1 = wx.BoxSizer(wx.VERTICAL)
sizer_1.Add(self.button_1, 0, 0, 0)
sizer_1.Add(self.panel_1, 1, wx.EXPAND, 0)
self.SetSizer(sizer_1)
self.Layout()
# end wxGlade
# end of class MyFrame
if __name__ == "__main__":
app = wx.PySimpleApp(0)
wx.InitAllImageHandlers()
frame_1 = MyFrame(None, -1, "")
app.SetTopWindow(frame_1)
frame_1.Show()
app.MainLoop()
好吧,我想这是我的功劳;但请注意:
- “wxMouseEvent不传播”
- “各种消息来源称(在实际的wxFrame元素中捕获鼠标事件)是不可能的,应该使用WxPanel来捕获事件”;“一些鼠标事件转到帧,如wxEVT_ENTER(或任何名称)。其他鼠标事件转到面板,如wxEVT_RIGHT_UP。您可以将这些事件转发到帧。”;鼠标事件不会在窗口层次结构中向上传播,因此,如果您的帧完全被其他窗口覆盖,那么它首先不会获取任何鼠标事件,因此您无法在那里捕获它们
- 每当您有wxFrame的一个子对象时,wxWidgets会自动假定您希望它覆盖wxFrame的整个区域。因此,框架的任何部分都不可见-因此,没有任何事件会将其发送给您的处理程序
- 递归地遍历框架中的所有(子)小部件李>
- 将鼠标左下/上事件单独连接到单个事件处理程序李>
- 确保单个事件处理程序以
结尾,以便将事件传播到原始按钮等,使其原始单击处理程序也能工作event.Skip()
#!/usr/bin/env python
# -*- coding: utf-8 -*-
# generated by wxGlade 0.6.3 on Mon Jan 12 21:20:22 2015
import wx
# begin wxGlade: extracode
# end wxGlade
class MyPanel(wx.Panel):
def __init__(self, parent, id=wx.ID_ANY, pos=wx.DefaultPosition, size=wx.DefaultSize, style=0, name="MyPanel"):
super(MyPanel, self).__init__(parent, id, pos, size, style, name)
self.parent = parent
self.lmb = False
self.Bind(wx.EVT_MOUSE_EVENTS, self.OnMouseEvents) # only reacts on its own clicks, not globally!
self.Bind(wx.EVT_PAINT, self.OnPaint)
def OnMouseEvents(self, event):
print("OnMouseEvents")
self.lmb = event.LeftDown() or event.LeftIsDown()
self.Refresh(False)
def OnPaint(self, event):
w, h = self.GetSize()
if self.lmb: mybrush = wx.Brush(wx.NamedColour("red"))
else: mybrush = wx.Brush(wx.NamedColour("blue"))
dc = wx.PaintDC(self)
dc.SetPen(wx.TRANSPARENT_PEN)
dc.SetBrush(mybrush)
dc.DrawRectangle(0, 0, w, h)
class MyFrame(wx.Frame):
def __init__(self, *args, **kwds):
# begin wxGlade: MyFrame.__init__
kwds["style"] = wx.DEFAULT_FRAME_STYLE
wx.Frame.__init__(self, *args, **kwds)
self.button_1 = wx.Button(self, -1, "button_1")
self.panel_1 = MyPanel(self, -1)
self.__set_properties()
self.__do_layout()
# end wxGlade
def __set_properties(self):
# begin wxGlade: MyFrame.__set_properties
self.SetTitle("frame_1")
self.SetSize((200, 200))
# end wxGlade
def __do_layout(self):
# begin wxGlade: MyFrame.__do_layout
sizer_1 = wx.BoxSizer(wx.VERTICAL)
sizer_1.Add(self.button_1, 0, 0, 0)
sizer_1.Add(self.panel_1, 1, wx.EXPAND, 0)
self.SetSizer(sizer_1)
self.Layout()
# end wxGlade
# end of class MyFrame
if __name__ == "__main__":
app = wx.PySimpleApp(0)
wx.InitAllImageHandlers()
frame_1 = MyFrame(None, -1, "")
app.SetTopWindow(frame_1)
frame_1.Show()
app.MainLoop()
#!/usr/bin/env python
# -*- coding: utf-8 -*-
# generated by wxGlade 0.6.3 on Mon Jan 12 21:20:22 2015
import wx
# begin wxGlade: extracode
# end wxGlade
class MyPanel(wx.Panel):
def __init__(self, parent, id=wx.ID_ANY, pos=wx.DefaultPosition, size=wx.DefaultSize, style=0, name="MyPanel"):
super(MyPanel, self).__init__(parent, id, pos, size, style, name)
self.lmb = False
#self.Bind(wx.EVT_MOUSE_EVENTS, self.OnMouseEvents) # only reacts on its own clicks, not globally! # use RecursiveSetListeners now; should be called from parent class after all init layout is done
self.Bind(wx.EVT_PAINT, self.OnPaint)
def OnMouseEvents(self, event):
print("OnMouseEvents")
self.lmb = event.LeftDown() or event.LeftIsDown()
self.Refresh(False)
event.Skip() # propagate, in case button is clicked
def OnPaint(self, event):
w, h = self.GetSize()
if self.lmb: mybrush = wx.Brush(wx.NamedColour("red"))
else: mybrush = wx.Brush(wx.NamedColour("blue"))
dc = wx.PaintDC(self)
dc.SetPen(wx.TRANSPARENT_PEN)
dc.SetBrush(mybrush)
dc.DrawRectangle(0, 0, w, h)
def RecursiveSetListeners(self, inwidget):
print("setting listeners: " + inwidget.GetName())
inwidget.Connect(wx.ID_ANY, -1, wx.wxEVT_LEFT_DOWN, self.OnMouseEvents)
inwidget.Connect(wx.ID_ANY, -1, wx.wxEVT_LEFT_UP, self.OnMouseEvents)
for child in inwidget.GetChildren():
self.RecursiveSetListeners(child)
class MyFrame(wx.Frame):
def __init__(self, *args, **kwds):
# begin wxGlade: MyFrame.__init__
kwds["style"] = wx.DEFAULT_FRAME_STYLE
wx.Frame.__init__(self, *args, **kwds)
self.button_1 = wx.Button(self, -1, "button_1", name="button_1")
self.panel_1 = MyPanel(self, -1, name="panel_1")
self.button_2 = wx.Button(self.panel_1, -1, "button_2", name="button_1")
#self.Connect(wx.ID_ANY, -1, wx.EVT_MOUSE_EVENTS, self.panel_1.OnMouseEvents) # 'wx.wxEVT_MOUSE_EVENTS: AttributeError: 'module' has no attribute 'wxEVT_MOUSE_EVENTS';; wx.EVT_MOUSE_EVENTS: TypeError: in method 'EvtHandler_Connect', expected argument 4 of type 'int'
# these work:
#self.Connect(wx.ID_ANY, -1, wx.wxEVT_LEFT_DOWN, self.panel_1.OnMouseEvents)
#self.Connect(wx.ID_ANY, -1, wx.wxEVT_LEFT_UP, self.panel_1.OnMouseEvents)
#self.button_1.Connect(wx.ID_ANY, -1, wx.wxEVT_LEFT_DOWN, self.panel_1.OnMouseEvents)
#self.button_1.Connect(wx.ID_ANY, -1, wx.wxEVT_LEFT_UP, self.panel_1.OnMouseEvents)
self.__set_properties()
self.__do_layout()
print(self.GetChildren())
self.panel_1.RecursiveSetListeners(self)
# end wxGlade
def __set_properties(self):
# begin wxGlade: MyFrame.__set_properties
self.SetTitle("frame_1")
self.SetSize((200, 200))
# end wxGlade
def __do_layout(self):
# begin wxGlade: MyFrame.__do_layout
sizer_2 = wx.BoxSizer(wx.HORIZONTAL)
sizer_2.Add(self.button_2, 0,0,0)
self.panel_1.SetSizer(sizer_2)
sizer_1 = wx.BoxSizer(wx.VERTICAL)
sizer_1.Add(self.button_1, 0, 0, 0)
sizer_1.Add(self.panel_1, 1, wx.EXPAND, 0)
self.SetSizer(sizer_1)
self.Layout()
# end wxGlade
# end of class MyFrame
if __name__ == "__main__":
app = wx.PySimpleApp(0)
wx.InitAllImageHandlers()
frame_1 = MyFrame(None, -1, "")
app.SetTopWindow(frame_1)
frame_1.Show()
app.MainLoop()
我更了解PyQt,但当您绑定并单击面板时,它只适用于面板本身。我不知道这是否可能,把它放在整个窗口上(这样按钮就会被整个窗口层覆盖)谢谢,@ProgrammingIsAwsome-没有意识到这一点,我只是找到了这样的链接,其中正好提到。。。干杯所以我不确定我只是在猜测,因为对我来说,这种叠加不会产生好的结果。我的意思是,那里的按钮会被覆盖,或者一次做多件事