使用看门狗和Tkinter

时间:2017-01-16 18:34:57

标签: python python-3.x user-interface tkinter

所以我有一个小脚本监视文件的更改/修改,如果文件被更改,它会像下面那样做一堆东西:

class Event(LoggingEventHandler):
def dispatch(self, event):
    #Do something here

if __name__ == "__main__":
    logging.basicConfig(level=logging.INFO,
                        format='%(asctime)s - %(message)s',
                        datefmt='%Y-%m-%d %H:%M:%S')
    event_handler = Event()
    observer = Observer()
    observer.schedule(event_handler, my_path, recursive=True)
    observer.start()
    try:
        while True:
            time.sleep(1)
    except KeyboardInterrupt:
        observer.stop()
    observer.join()

它很棒。但我有另一个脚本与一堆Tkinter小部件(典型的tk app与mainloop())。现在,当用户按下某个按钮时,我想调用监视程序来监视文件更改,并调用dispatch(),它会对它执行一些操作,并在同一个脚本中更新一些tkinter ui小部件,例如progressbars。

这个GUI的新东西,无法弄清楚如何使看门狗循环和root.mainloop一起工作。怎么办呢?

1 个答案:

答案 0 :(得分:6)

Watchdog在自己的线程中运行,因此您不需要做太多事情。如果要根据事件修改GUI,则应设置线程安全队列。 Tkinter小部件不应该由多个线程修改,因此常见的模式是使用线程安全队列在线程之间进行通信。

以下示例在队列中放置一个监视程序事件,并在检测到监视程序事件时使用event_generate将信号从观察者发送到GUI。我绝对不确定以下内容在所有情况下都能正常工作,因为我现在只能访问一个linux盒来测试它。但是,它似乎在Linux上运行正常。

import Tkinter as tk
from watchdog.observers import Observer
from watchdog.events import FileSystemEventHandler

from Queue import Queue
import sys

class CustomHandler(FileSystemEventHandler):
    def __init__(self, app):
        FileSystemEventHandler.__init__(self)
        self.app = app
    def on_created(self, event): app.notify(event)
    def on_deleted(self, event): app.notify(event)
    def on_modified(self, event): app.notify(event)
    def on_moved(self, event): app.notify(event)

class App(object):
    def __init__(self):
        path = sys.argv[1] if len(sys.argv) > 1 else "/tmp"
        handler = CustomHandler(self)
        self.observer = Observer()
        self.observer.schedule(handler, path, recursive=True)

        self.queue = Queue()
        self.root = tk.Tk()

        self.text = tk.Text(self.root)
        self.text.pack(fill="both", expand=True)

        self.text.insert("end", "Watching %s...\n" % path)

        self.root.bind("<Destroy>", self.shutdown)
        self.root.bind("<<WatchdogEvent>>", self.handle_watchdog_event)

        self.observer.start()

    def handle_watchdog_event(self, event):
        """Called when watchdog posts an event"""
        watchdog_event = self.queue.get()
        print("event type:", type(watchdog_event))
        self.text.insert("end", str(watchdog_event) + "\n")

    def shutdown(self, event):
        """Perform safe shutdown when GUI has been destroyed"""
        self.observer.stop()
        self.observer.join()

    def mainloop(self):
        """Start the GUI loop"""
        self.root.mainloop()

    def notify(self, event):
        """Forward events from watchdog to GUI"""
        self.queue.put(event)
        self.root.event_generate("<<WatchdogEvent>>", when="tail")

app = App()
app.mainloop()