gpt4 book ai didi

python - Tkinter 启动画面和主循环外的多处理

转载 作者:行者123 更新时间:2023-12-03 21:29:45 24 4
gpt4 key购买 nike

我已经实现了一个启动画面,当我的应用程序在启动时从远程云存储加载数据库时显示。启动画面通过调用 .update() 保持事件状态(上面有一个进度条),并在单独的加载过程结束后销毁。在此之后,主循环启动并且应用程序正常运行。

下面的代码曾经在我的 Mac 上使用 python 3.6 和 tcl/tk 8.5.9 运行良好。但是,在更新到 Sierra 之后,我被迫将 tk 更新到 ActiveTcl 8.5.18。现在,在单独的进程完成之前不会显示初始屏幕,但随后会与根窗口一起出现并停留在屏幕上(即使调用了其 .destroy() 方法)。

import tkinter as tk
import tkinter.ttk as ttk
import multiprocessing
import time


class SplashScreen(tk.Toplevel):
def __init__(self, root):
tk.Toplevel.__init__(self, root)
self.geometry('375x375')
self.overrideredirect(True)

self.columnconfigure(0, weight=1)
self.rowconfigure(0, weight=1)

self.label = ttk.Label(self, text='My Splashscreen', anchor='center')
self.label.grid(column=0, row=0, sticky='nswe')

self.center_splash_screen()
print('initialized splash')

def center_splash_screen(self):
w = self.winfo_screenwidth()
h = self.winfo_screenheight()
x = w / 2 - 375 / 2
y = h / 2 - 375 / 2
self.geometry("%dx%d+%d+%d" % ((375, 375) + (x, y)))

def destroy_splash_screen(self):
self.destroy()
print('destroyed splash')


class App(tk.Tk):
def __init__(self):
tk.Tk.__init__(self)

self.start_up_app()

self.title("MyApp")
self.columnconfigure(0, weight=1)
self.rowconfigure(0, weight=1)

self.application_frame = ttk.Label(self, text='Rest of my app here', anchor='center')
self.application_frame.grid(column=0, row=0, sticky='nswe')

self.mainloop()

def start_up_app(self):
self.show_splash_screen()

# load db in separate process
process_startup = multiprocessing.Process(target=App.startup_process)
process_startup.start()

while process_startup.is_alive():
# print('updating')
self.splash.update()

self.remove_splash_screen()

def show_splash_screen(self):
self.withdraw()
self.splash = SplashScreen(self)

@staticmethod
def startup_process():
# simulate delay while implementation is loading db
time.sleep(5)

def remove_splash_screen(self):
self.splash.destroy_splash_screen()
del self.splash
self.deiconify()

if __name__ == '__main__':
App()

我不明白为什么会发生这种情况以及如何解决它。有人可以帮忙吗?谢谢!

更新:

如果您注释掉 self.overrideredirect(True) 行,启动画面将正确显示.但是,我不想要窗口装饰,它在脚本结束时仍然保留在屏幕上。它正在内部销毁,任何进一步的方法调用 self.splash (例如 .winfo_... -methods)导致 _tkinter.TclError: bad window path name ".!splashscreen" .

此外,此代码在 windows 和 tcl/tk 8.6 下运行良好。这是 Mac 上 tcl/tk 8.5.18 窗口管理的错误/问题吗?

最佳答案

显然,这是由于在调用 overrideredirect(True) 后窗口管理器未装饰窗口时窗口堆叠顺序存在问题。 .它似乎也发生在其他平台上。

在使用 Python 3.6.1 和 tcl/tk 8.5.18 的 macOS 10.12.5 上运行以下代码,单击“打开”按钮后不会出现顶级窗口:

import tkinter as tk

class TL(tk.Toplevel):
def __init__(self):
tk.Toplevel.__init__(self)
self.overrideredirect(True)
# self.after_idle(self.lift)
tl_label = tk.Label(self, text='this is a undecorated\ntoplevel window')
tl_label.grid(row=0)
b_close = tk.Button(self, text='close', command=self.close)
b_close.grid(row=1)

def close(self):
self.destroy()

def open():
TL()

root = tk.Tk()
label = tk.Label(root, text='This is the root')
label.grid(row=0)
b_open = tk.Button(root, text='open', command=open)
b_open.grid(row=1)
root.mainloop()

取消注释行 self.after_idle(self.lift)解决了这个问题(只需调用 self.lift() 也可以。但是使用 after_idle() 可以防止窗口在移动到其位置并调整大小之前闪烁几分之一秒,这是我在使用 tkinter 和让我想知道我是否应该继续学习 PyQT 或 PySide2 ......)。

至于在我的原始问题中关闭未装饰窗口的问题:调用 after_idle(window.destroy()) 而不是 window.destroy() 似乎也解决了这个问题。我不理解为什么。

万一其他人重现这个并且有人提示我在哪里将其报告为错误,我很乐意这样做。

关于python - Tkinter 启动画面和主循环外的多处理,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/44802456/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com