Tkinter 进度条在 Windows 下不工作

Tkinter progressbar not working under Windows

我正在编写一个将在 Linux 和 Windows 下使用的应用程序。

对于 GUI,我使用的是 Tkinter,但是当我尝试在 Windows 上 运行 我的程序时遇到了问题。在我开发它的 Linux 上,它工作得很好。

在 program/GUI 出现之前,我使用了带有 ttk 进度条的顶层来显示加载数据的进度。 Windows 上缺少此进度条。它不会引发错误,它似乎不存在。它应该在的地方是透明的,所以你可以看到顶层后面的 windows/desktop 等。

知道为什么会这样吗?还有其他人遇到过这个问题吗?

调用顶层的代码如下:

import Tkinter as tk
import ttk
import platform
import time

class Window(tk.Tk):
    def __init__(self):
        tk.Tk.__init__(self)
        self.init_progbar()
        self.prog_data()
        self.top.destroy()

    #initializing progress bar, may be used with self.top
    def init_progbar(self):
        self.top = tk.Toplevel(self)
        self.top.frame = tk.Frame(self.top)
        self.top.frame.grid()
        self.top.frame.progbar = ttk.Progressbar(self.top.frame, \
            orient='horizontal', mode = 'determinate', length = '500')
        self.top.frame.progbar.grid()
        Window.center(self.top)

    #static method to center a toplevel window
    @staticmethod
    def center(toplevel, size=None):
        toplevel.update_idletasks()
        w = toplevel.winfo_screenwidth()
        h = toplevel.winfo_screenheight()
        if not size:
            size = tuple(int(_) for _ in toplevel.geometry().split('+')[0].split('x'))
        x = w/2 - size[0]/2
        y = h/2 - size[1]/2
        toplevel.geometry("%dx%d+%d+%d" % (size + (x, y)))

    #dummy method - somethings done, then the value of the progress bar
    #is raised by 10
    def prog_data(self):
        for i in range(10):
            time.sleep(1)
            self.top.frame.progbar.step(10)
            self.top.update_idletasks()

def main():
    window = Window()
    window.mainloop()

if __name__=='__main__':
    main()

正如我之前所说,这是 运行ning 在 Linux 下没有任何问题,只是在 Windows 上不起作用。

我在两个不同的 Linux 系统(openSUSE 和 Ubuntu)和两台不同的计算机 运行ning Windows,一个 Windows 7和一个 Windows 10. 每次出现同样的问题 - Windows.

上缺少进度条

自己找到解决办法。问题是 Windows 和 Linux 下的执行似乎有点不同。

Windows 在您进入上例中的主循环之前不会更新 GUI。此事在Linux上的处理明显不同,保证正确执行。

为了清楚起见,我重新编写了示例代码,这样可能会更清楚需要更改的内容。查看评论以获得进一步的解释。

import Tkinter as tk
import ttk
import platform
import time

class Window(tk.Tk):
    def __init__(self):
        tk.Tk.__init__(self)
        self.init_progbar()
        self.prog_data()
        self.top.destroy()

    #initializing progress bar, may be used with self.top.
    def init_progbar(self):
        #The frame here was unnecessary.
        self.top = tk.Toplevel(self)
        self.top.progbar = ttk.Progressbar(self.top, \
            orient='horizontal', mode = 'determinate', length = '500')
        self.top.progbar.grid()
        Window.center(self.top)

    #static method to center a toplevel window
    @staticmethod
    def center(toplevel, size=None):
        toplevel.update_idletasks()
        w = toplevel.winfo_screenwidth()
        h = toplevel.winfo_screenheight()
        if not size:
            size = tuple(int(_) for _ in toplevel.geometry().split('+')[0].split('x'))
        x = w/2 - size[0]/2
        y = h/2 - size[1]/2
        toplevel.geometry("%dx%d+%d+%d" % (size + (x, y)))
        #This processes every pending geometry change and redraws the widget.
        self.update()

    #dummy method - somethings done, then the value of the progress bar
    #is raised by 10
    def prog_data(self):
        for i in range(10):
            time.sleep(1)
            self.top.frame.progbar.step(10)
            #Updates the whole GUI every time some data gets processed.
            self.update()

def main():
    window = Window()
    window.mainloop()

if __name__=='__main__':
    main()