wxPython:在其他线程中执行某些操作时显示框架

时间:2014-04-30 15:45:35

标签: python multithreading wxpython

在我的GUI中使用wxPython,如果必须进行一些可能需要一些时间的计算。所以我想在一个单独的线程中启动它们,并在GUI中显示一个窗口,用于打印程序正在计算的内容。在此期间应禁用主窗口。 这就是我的代码:

import time
import threading

import wx

def calculate():
    # Simulates the calculation
    time.sleep(5)
    return True

class CalcFrame(wx.Frame):

    def __init__(self, parent, id):
        wx.Frame.__init__(self, parent=None, id=-1, title="Calculate")
        # Normally here are some intctrls, but i dont show them to keep it easy
        self.panel = wx.Panel(parent=self, id=-1)
        self.createButtons()

    def createButtons(self):
        button = wx.Button(parent=self.panel, id=-1, label="Calculate")
        button.Bind(wx.EVT_BUTTON, self.onCalculate)

    def onCalculate(self, event):
        calcThread = threading.Thread(target=calculate)
        checkThread = threading.Thread(target=self.checkThread, args=(calcThread,))
        self.createWaitingFrame()
        self.waitingFrame.Show(True)
        self.Disable()
        calcThread.run()
        checkThread.run()

    def createWaitingFrame(self):
        self.waitingFrame = wx.MiniFrame(parent=self, title="Please wait")
        panel = wx.Panel(parent=self.waitingFrame)
        waitingText = wx.StaticText(parent=panel, label="Please wait - Calculating", style=wx.ALIGN_CENTER)

    def checkThread(self, thread):
        while thread.is_alive():
            pass
        print "Fertig"
        self.waitingFrame.Destroy()
        self.Enable()

app = wx.PySimpleApp()
frame = CalcFrame(parent=None, id=-1)
frame.Show()
app.MainLoop()

但我现在的问题是,如果我按下"计算"按钮,waitingFrame没有显示正确,我看不到文字。我也无法移动/最大化/最小化主窗口。 你能帮助我吗?提前谢谢你:)

1 个答案:

答案 0 :(得分:1)

你永远不应该在主线程以外的线程中更新gui ....我很确定wxPython的文档在几个地方提到它..它不是线程安全的,你的gui处于破碎的状态......

相反,我认为你应该做像

这样的事情
def thread1():
     time.sleep(5)
     return True

def thread2(t1,gui):
    while thread.is_alive():
        pass
    print "Fertig"
    wx.CallAfter(gui.ThreadDone)

class MyFrame(wx.Frame):
   def startThread(self):
      calcThread = threading.Thread(target=thread1)
      checkThread = threading.Thread(target=thread2, args=(calcThread,self))
   def ThreadDone(self):
       print "Both threads done???"
       print "Now modify gui from main thread(here!)"