如何使用 tkinter 事件“继续”或暂停不同的 while 循环?

Posted

技术标签:

【中文标题】如何使用 tkinter 事件“继续”或暂停不同的 while 循环?【英文标题】:How do I use tkinter events to 'continue' or pause a different while loop? 【发布时间】:2020-10-23 09:23:31 【问题描述】:

我在主线程上运行了 tkinter GUI,并在第二个线程上运行了一个 while 循环,该循环更新了要在 GUI 上显示的标签内容,持续时间为 n 秒。这本身运行良好。 现在我想在 GUI 上添加一个 > 按钮来进行循环:

暂停当前迭代,使用上一项调用显示方法,完成后继续迭代 中途跳过当前项目并移至下一个项目。 但我找不到如何生成循环中断事件。我不能根据按钮生成的某些变量/db 条目调用 continue 或 sleep(比如这里的 Python: Tkinter pausing a while loop),因为中断可能发生在循环中的任何地方(而不是特定的 if/else 行)

我已经通过 codemy.com - https://github.com/flatplanet/Intro-To-TKinter-Youtube-Course/blob/master/viewer.py 和谷歌提到了 Viewer 程序。

这是我正在尝试做的事情。 (https://gitlab.com/ananya26nov/PixelescoPy)

主线程运行 GUI

guiObj = GuiWindow()
thread_list = []
thread = threading.Thread(target=worker, args=(guiObj,))
thread_list.append(thread)
thread.start()
guiObj.run_window_on_loop()
thread.join()
guiObj.quit_window()

线程的worker方法是这样的:

 while len(files_not_viewed) != 0:
    chosen = random.choice(files_not_viewed)
    if is_not_viewed(chosen):
        pictureObj = Picture(chosen)
        # Display the chosen for <timer> seconds
        pictureObj.display(guiObj)
        time.sleep(timer)
        # Change the status of "chosen" to "viewed"
        mark_as_viewed(chosen)
    files_not_viewed = list(set(files_not_viewed) - set([chosen]))

显示方法调用下面类的'add_input_section'方法

class GuiWindow():
def __init__(self):
    self.root = Tk()
    self.screen_width = self.root.winfo_screenwidth()
    self.screen_height = self.root.winfo_screenheight()
    self.image_label = None
    self.image = None
    self.folder_path = None
    self.timer = None

def run_window_on_loop(self):
    button_exit = Button(self.root, text="Exit Program", command=self.root.quit)
    button_exit.grid(row=1, column=1)
    self.root.mainloop()

def quit_window(self):
    self.root.quit()

def resize(self, image_path):
    my_pic = Image.open(image_path)
    pic_height = my_pic.height
    pic_width = my_pic.width
    if my_pic.height > (self.screen_height - 100):
        new_height= self.screen_height - 100
        new_width = int(new_height / pic_height * pic_width)
        pic_height = new_height
        pic_width = new_width

    if pic_width > self.screen_width - 5:
        new_width = self.screen_height - 5
        new_height = int(new_width / pic_width * pic_height)
        pic_height = new_height
        pic_width = new_width

    resized_image = my_pic.resize((pic_width, pic_height), Image.ANTIALIAS)
    return resized_image

def add_image(self, image_path):
    resized_img = self.resize(image_path)
    image_obj = ImageTk.PhotoImage(resized_img)
    image_label = Label(self.root, image=image_obj,
                        height=resized_img.height,
                        width=resized_img.width)
    self.image = image_obj # DO NOT REMOVE - Garbage collector error
    if self.image_label is not None:
        self.remove_image()
    image_label.grid(row=0, column=0, columnspan=3)
    self.image_label = image_label

def remove_image(self):
    self.image_label.grid_forget()

【问题讨论】:

如果没有所有代码就很难判断,但您似乎正在从另一个线程操作 GUI,这通常是不允许的。在担心如何在线程之间进行通信之前,您可能需要重新处理它。 @DavisHerring 是的,我有一个线程正在操作 GUI。所以这很好。我可以看到这个线程控制的新图像。现在我想要 GUI 中的一个按钮,它能够暂停线程的循环。如果您想看一下,代码就在这里:/:gitlab.com/ananya26nov/PixelescoPy 【参考方案1】:

使用 tk.BooleanVar() 来操作变量,并在您尝试暂停时跟踪状态,以便在完成后恢复。

【讨论】:

以上是关于如何使用 tkinter 事件“继续”或暂停不同的 while 循环?的主要内容,如果未能解决你的问题,请参考以下文章

Tkinter - 如何在事件发生时添加“铃声”?

在Python中触发来自不同类的Tkinter事件

问下:在Tkinter中如何使程序暂停几秒钟再执行

在python tkinter中为事件创建日历视图[关闭]

如何使点击事件通过tkinter窗口?

tkinter窗口鼠标拖动就会暂停