mirror of
https://github.com/wxWidgets/Phoenix.git
synced 2025-07-21 04:31:09 +02:00
In Python 3.10, a change[1] was implemented where extension functions that take integer arguments will no longer silently accept non-integer arguments (e.g., floats) that can only be converted to integers with a loss of precision. This PR fixes most of these issues in the pure-Python classes and demos by explicitly converting the parameters to int before passing them to wxWidgets. There is loss of precision, but this was happening before (automatically) anyway as most wxWidgets DeviceContext functions operate using integers. Additionally, the PR fixes a few sizing issues, mostly with SpinCtrls being too small on GTK3. This is an example of the relevant exception: Traceback (most recent call last): File "/usr/lib64/python3.10/site-packages/wx/lib/agw/pygauge.py", line 355, in OnPaint r.width = w TypeError: 'float' object cannot be interpreted as an integer Fixes #2038. [1] https://bugs.python.org/issue37999
228 lines
7.3 KiB
Python
228 lines
7.3 KiB
Python
#!/usr/bin/env python
|
|
|
|
# 11/30/2003 - Jeff Grimmett (grimmtooth@softhome.net)
|
|
#
|
|
# o Some issues with the listbox example; I tried correcting
|
|
# it but it's still not working the way it should. Commented
|
|
# out for now, as I found it.
|
|
|
|
|
|
import wx
|
|
|
|
#---------------------------------------------------------------------------
|
|
|
|
class TestPopup(wx.PopupWindow):
|
|
"""Adds a bit of text and mouse movement to the wx.PopupWindow"""
|
|
def __init__(self, parent, style):
|
|
wx.PopupWindow.__init__(self, parent, style)
|
|
pnl = self.pnl = wx.Panel(self)
|
|
pnl.SetBackgroundColour("CADET BLUE")
|
|
|
|
|
|
st = wx.StaticText(pnl, -1,
|
|
"This is a special kind of top level\n"
|
|
"window that can be used for\n"
|
|
"popup menus, combobox popups\n"
|
|
"and such.\n\n"
|
|
"Try positioning the demo near\n"
|
|
"the bottom of the screen and \n"
|
|
"hit the button again.\n\n"
|
|
"In this demo this window can\n"
|
|
"be dragged with the left button\n"
|
|
"and closed with the right."
|
|
,
|
|
pos=(10,10))
|
|
|
|
sz = st.GetBestSize()
|
|
self.SetSize( (sz.width+20, sz.height+20) )
|
|
pnl.SetSize( (sz.width+20, sz.height+20) )
|
|
|
|
pnl.Bind(wx.EVT_LEFT_DOWN, self.OnMouseLeftDown)
|
|
pnl.Bind(wx.EVT_MOTION, self.OnMouseMotion)
|
|
pnl.Bind(wx.EVT_LEFT_UP, self.OnMouseLeftUp)
|
|
pnl.Bind(wx.EVT_RIGHT_UP, self.OnRightUp)
|
|
|
|
st.Bind(wx.EVT_LEFT_DOWN, self.OnMouseLeftDown)
|
|
st.Bind(wx.EVT_MOTION, self.OnMouseMotion)
|
|
st.Bind(wx.EVT_LEFT_UP, self.OnMouseLeftUp)
|
|
st.Bind(wx.EVT_RIGHT_UP, self.OnRightUp)
|
|
|
|
wx.CallAfter(self.Refresh)
|
|
|
|
|
|
def OnMouseLeftDown(self, evt):
|
|
self.Refresh()
|
|
self.ldPos = evt.GetEventObject().ClientToScreen(evt.GetPosition())
|
|
self.wPos = self.ClientToScreen((0,0))
|
|
self.pnl.CaptureMouse()
|
|
|
|
def OnMouseMotion(self, evt):
|
|
if evt.Dragging() and evt.LeftIsDown():
|
|
dPos = evt.GetEventObject().ClientToScreen(evt.GetPosition())
|
|
nPos = (self.wPos.x + (dPos.x - self.ldPos.x),
|
|
self.wPos.y + (dPos.y - self.ldPos.y))
|
|
self.Move(nPos)
|
|
|
|
def OnMouseLeftUp(self, evt):
|
|
if self.pnl.HasCapture():
|
|
self.pnl.ReleaseMouse()
|
|
|
|
def OnRightUp(self, evt):
|
|
self.Show(False)
|
|
wx.CallAfter(self.Destroy)
|
|
|
|
|
|
|
|
class TestTransientPopup(wx.PopupTransientWindow):
|
|
"""Adds a bit of text and mouse movement to the wx.PopupWindow"""
|
|
def __init__(self, parent, style, log):
|
|
wx.PopupTransientWindow.__init__(self, parent, style)
|
|
self.log = log
|
|
panel = wx.Panel(self)
|
|
panel.SetBackgroundColour("#FFB6C1")
|
|
|
|
st = wx.StaticText(panel, -1,
|
|
"wx.PopupTransientWindow is a\n"
|
|
"wx.PopupWindow which disappears\n"
|
|
"automatically when the user\n"
|
|
"clicks the mouse outside it or if it\n"
|
|
"(or its first child) loses focus in \n"
|
|
"any other way.")
|
|
btn = wx.Button(panel, -1, "Press Me")
|
|
spin = wx.SpinCtrl(panel, -1, "Hello")
|
|
btn.Bind(wx.EVT_BUTTON, self.OnButton)
|
|
|
|
sizer = wx.BoxSizer(wx.VERTICAL)
|
|
sizer.Add(st, 0, wx.ALL, 5)
|
|
sizer.Add(btn, 0, wx.ALL, 5)
|
|
sizer.Add(spin, 0, wx.ALL, 5)
|
|
panel.SetSizer(sizer)
|
|
|
|
sizer.Fit(panel)
|
|
sizer.Fit(self)
|
|
self.Layout()
|
|
|
|
|
|
def ProcessLeftDown(self, evt):
|
|
self.log.write("ProcessLeftDown: %s\n" % evt.GetPosition())
|
|
return wx.PopupTransientWindow.ProcessLeftDown(self, evt)
|
|
|
|
def OnDismiss(self):
|
|
self.log.write("OnDismiss\n")
|
|
|
|
def OnButton(self, evt):
|
|
btn = evt.GetEventObject()
|
|
if btn.GetLabel() == "Press Me":
|
|
btn.SetLabel("Pressed")
|
|
else:
|
|
btn.SetLabel("Press Me")
|
|
|
|
|
|
|
|
class TestPanel(wx.Panel):
|
|
def __init__(self, parent, log):
|
|
wx.Panel.__init__(self, parent, -1)
|
|
self.log = log
|
|
|
|
b = wx.Button(self, -1, "Show wx.PopupWindow", (25, 50))
|
|
self.Bind(wx.EVT_BUTTON, self.OnShowPopup, b)
|
|
|
|
b = wx.Button(self, -1, "Show wx.PopupTransientWindow", (25, 95))
|
|
self.Bind(wx.EVT_BUTTON, self.OnShowPopupTransient, b)
|
|
|
|
# This isn't working so well, not sure why. Commented out for
|
|
# now.
|
|
|
|
# b = wx.Button(self, -1, "Show wx.PopupWindow with listbox", (25, 140))
|
|
# self.Bind(wx.EVT_BUTTON, self.OnShowPopupListbox, b)
|
|
|
|
|
|
def OnShowPopup(self, evt):
|
|
win = TestPopup(self.GetTopLevelParent(), wx.SIMPLE_BORDER)
|
|
#win = TestPopupWithListbox(self, wx.SIMPLE_BORDER, self.log)
|
|
|
|
# Show the popup right below or above the button
|
|
# depending on available screen space...
|
|
btn = evt.GetEventObject()
|
|
pos = btn.ClientToScreen( (0,0) )
|
|
sz = btn.GetSize()
|
|
win.Position(pos, (0, sz[1]))
|
|
|
|
win.Show(True)
|
|
|
|
|
|
def OnShowPopupTransient(self, evt):
|
|
win = TestTransientPopup(self,
|
|
wx.SIMPLE_BORDER,
|
|
self.log)
|
|
|
|
# Show the popup right below or above the button
|
|
# depending on available screen space...
|
|
btn = evt.GetEventObject()
|
|
pos = btn.ClientToScreen( (0,0) )
|
|
sz = btn.GetSize()
|
|
win.Position(pos, (0, sz[1]))
|
|
|
|
win.Popup()
|
|
|
|
|
|
def OnShowPopupListbox(self, evt):
|
|
win = TestPopupWithListbox(self, wx.NO_BORDER, self.log)
|
|
|
|
# Show the popup right below or above the button
|
|
# depending on available screen space...
|
|
btn = evt.GetEventObject()
|
|
pos = btn.ClientToScreen( (0,0) )
|
|
sz = btn.GetSize()
|
|
win.Position(pos, (0, sz[1]))
|
|
|
|
win.Show(True)
|
|
|
|
|
|
|
|
# This class is currently not implemented in the demo. It does not
|
|
# behave the way it should, so for the time being it's only here
|
|
# for show. If you figure out how to make it work, please send
|
|
# a corrected file to Robin!
|
|
class TestPopupWithListbox(wx.PopupWindow):
|
|
def __init__(self, parent, style, log):
|
|
wx.PopupWindow.__init__(self, parent, style)
|
|
self.log = log
|
|
import keyword
|
|
self.lb = wx.ListBox(self, -1, choices = keyword.kwlist)
|
|
#sz = self.lb.GetBestSize()
|
|
self.SetSize((150, 75)) #sz)
|
|
self.lb.SetSize(self.GetClientSize())
|
|
self.lb.SetFocus()
|
|
self.Bind(wx.EVT_LISTBOX, self.OnListBox)
|
|
self.Bind(wx.EVT_LISTBOX_DCLICK, self.OnListBoxDClick)
|
|
|
|
def OnListBox(self, evt):
|
|
obj = evt.GetEventObject()
|
|
self.log.write("OnListBox: %s\n" % obj)
|
|
self.log.write('Selected: %s\n' % obj.GetString(evt.GetInt()))
|
|
evt.Skip()
|
|
|
|
def OnListBoxDClick(self, evt):
|
|
self.Hide()
|
|
self.Destroy()
|
|
|
|
#---------------------------------------------------------------------------
|
|
|
|
def runTest(frame, nb, log):
|
|
win = TestPanel(nb, log)
|
|
return win
|
|
|
|
#---------------------------------------------------------------------------
|
|
|
|
|
|
overview = """\
|
|
"""
|
|
|
|
|
|
if __name__ == '__main__':
|
|
import sys,os
|
|
import run
|
|
run.main(['', os.path.basename(sys.argv[0])] + sys.argv[1:])
|
|
|