通过新线程在Python中打开wx.Frame

0 投票
1 回答
1533 浏览
提问于 2025-04-15 13:23

我有一个框架,它作为用户在主程序启动之前进行选择的启动屏幕。在用户做出选择后,我希望这个屏幕能保持显示,就像一个欢迎界面,直到主程序在后台加载完成。

我通过创建一个应用程序并启动一个线程来实现这个功能:

class App(wx.App):
    '''
    Creates the main frame and displays it
    Returns true if successful
    '''
    def OnInit(self):
        try:
            '''
            Initialization
            '''
            self.newFile = False
            self.fileName = ""

            self.splashThread = Splash.SplashThread(logging, self)
            self.splashThread.start()
            #...More to the class

这个线程会启动一个框架:

class SplashThread(threading.Thread):
    def __init__(self, logger, app):
        threading.Thread.__init__(self)
        self.logger = logger
        self.app = app

    def run(self):
        frame = Frame(self.logger, self.app)
        frame.Show()

这里的app值是必要的,因为它包含了一个回调函数,可以在用户做出选择后让主程序继续运行。问题是,启动屏幕只闪现了毫秒级的时间就消失了,这样用户根本没法做选择,导致整个启动过程被阻塞。

有什么好的建议吗?提前谢谢大家!

1 个回答

0

你不需要用线程来解决这个问题。缺点是加载的时候,启动画面会被阻塞,也就是说在加载的时候你不能更新它的内容(比如动画效果)或者不能拖动它。如果你想解决这个问题,可以定期调用一下 wx.SafeYield 之类的函数。

import time
import wx


class Loader(wx.Frame):
    def __init__(self):
        wx.Frame.__init__(self, None)
        sizer = wx.BoxSizer(wx.VERTICAL)
        self.SetSizer(sizer)
        self.btn1 = wx.Button(self, label="Option 1")
        self.btn2 = wx.Button(self, label="Option 2")
        sizer.Add(self.btn1, flag=wx.EXPAND)
        sizer.Add(self.btn2, flag=wx.EXPAND)
        self.btn1.Bind(wx.EVT_BUTTON, self.OnOption1)
        self.btn2.Bind(
            wx.EVT_BUTTON, lambda e: wx.MessageBox("There is no option 2")
        )

    def OnOption1(self, event):
        self.btn1.Hide()
        self.btn2.Hide()
        self.Sizer.Add(
            wx.StaticText(self, label="Loading Option 1..."),
            1, wx.ALL | wx.EXPAND, 15
        )
        self.Layout()
        self.Update()
        AppFrame(self).Show()

class AppFrame(wx.Frame):
    def __init__(self, parent):
        wx.Frame.__init__(self, parent)
        time.sleep(3)
        parent.Hide()

        # the top window (Loader) is hidden so the app needs to be told to exit
        # when this window is closed
        self.Bind(wx.EVT_CLOSE, lambda e: wx.GetApp().ExitMainLoop())


app = wx.PySimpleApp()
app.TopWindow = Loader()
app.TopWindow.Show()
app.MainLoop()

撰写回答