我正在为我的应用程序构建一个面板,其中包含大量缩略图图像;它们不会全部适合窗口,所以我使用ScrolledWindow来保存它们并允许用户滚动查看它们。
生成缩略图图像需要一些时间,因此我尝试为窗口中当前可见的缩略图优先生成图像。 我还想捕获哪些缩略图是可见的,以便如果我添加或删除缩略图的子集,我知道滚动条的位置以包含尽可能多的以前可见的缩略图。
我遇到的问题是,在子窗口重新定位在 ScrolledWindow 的工作区之前,EVT_SCROLLWIN似乎是发布的;换句话说,我正在测量滚动条移动之前的可见对象。 有没有办法:a( 强制 ScrolledWindow 更新子窗口位置,b( 可靠地确定新客户端窗口偏移量与现在相比的偏移量,或 c( 在此更新发生后调用事件处理程序?
如果重要的话,我正在这台计算机上运行Python 2.7.11和wxPython 3.0.2.0。
代码段如下:
class ThumbsViewer(wx.ScrolledWindow):
def __init__(self, parent, info, style = wx.BORDER_RAISED | wx.HSCROLL):
wx.ScrolledWindow.__init__(self, parent = parent, id = wx.ID_ANY, size = wx.DefaultSize,
pos = wx.DefaultPosition, style = style, name = "ThumbsViewer")
self.__info__ = info
self.__bitmapctrls__ = []
self.__selected__ = []
self.__lastclicked__ = []
self.Bind(wx.EVT_SCROLLWIN, self.__OnScroll__)
# ... Code to generate placeholder thumbnails, arrange buttons, size window, etc. ...
# ... Various functions to handle clicking on thumbnails, etc. ...
# EVT_SCROLLWIN Handler - Measure client window positions and determine which are
# visible on the screen.
def __OnScroll__(self, event):
if event:
event.Skip()
priority = []
clientsize = self.GetClientSize()
for ctrl in self.__bitmapctrls__:
pos = ctrl.GetPosition()
size = ctrl.GetSize()
# Test to see if any part of the thumbnail control falls within the client
# area...append to the priority list if it does.
# This appears to be where the problem is - the child window positions
# have not been updated to reflect the new scrollbar position yet.
if (((pos[0] >= 0 and pos[0] <= clientsize[0]) or (pos[0] + size[0] >= 0 and pos[0] + size[0] <= clientsize[0]))
and ((pos[1] >= 0 and pos[1] <= clientsize[1]) or (pos[1] + size[1] >= 0 and pos[1] + size[1] <= clientsize[1]))):
priority.append(ctrl.GetPage())
self.__info__.SetPriorityPages(priority)
在为此苦苦挣扎了一段时间后,我决定绑定EVT_PAINT而不是EVT_SCROLLWIN,并在发生这种情况时进行滚动条检查。 重新绘制发生在EVT_SCROLLWIN和子窗口位置更新后。
我对这个解决方案不是很满意,因为EVT_PAINT发生的原因有很多与滚动条无关,所以如果您有更好的解决方案,请随时指出更好的解决方案。 幸运的是,在这种情况下,计算很快就会失败,除非我添加或删除缩略图(大多数绘制事件不会发生这种情况(,因此对性能的影响很小。