Перенаправить результаты командной строки в графический интерфейс tkinter - PullRequest
21 голосов
/ 20 марта 2009

Я создал программу, которая печатает результаты в командной строке. (Это сервер, и он печатает журнал в командной строке.)

Теперь я хочу увидеть тот же результат в графическом интерфейсе.

Как перенаправить результаты командной строки в графический интерфейс?

Пожалуйста, предложите хитрость, чтобы легко преобразовать консольное приложение в простой графический интерфейс.

Обратите внимание, что он должен работать в Linux и Windows.

Ответы [ 4 ]

13 голосов
/ 20 марта 2009

Вы можете создать оболочку сценария, которая запускает вашу программу командной строки как подпроцесс, а затем добавить вывод в нечто вроде текстового виджета.

from tkinter import *
import subprocess as sub
p = sub.Popen('./script',stdout=sub.PIPE,stderr=sub.PIPE)
output, errors = p.communicate()

root = Tk()
text = Text(root)
text.pack()
text.insert(END, output)
root.mainloop()

где скрипт - это ваша программа. Очевидно, вы можете распечатать ошибки другим цветом или чем-то в этом роде.

11 голосов
/ 20 сентября 2015

Чтобы отобразить вывод подпроцесса в графическом интерфейсе , пока он все еще работает , портативное решение только для stdlib, которое работает как на Python 2, так и 3, должно использовать фоновый поток:

#!/usr/bin/python
"""
- read output from a subprocess in a background thread
- show the output in the GUI
"""
import sys
from itertools import islice
from subprocess import Popen, PIPE
from textwrap import dedent
from threading import Thread

try:
    import Tkinter as tk
    from Queue import Queue, Empty
except ImportError:
    import tkinter as tk # Python 3
    from queue import Queue, Empty # Python 3

def iter_except(function, exception):
    """Works like builtin 2-argument `iter()`, but stops on `exception`."""
    try:
        while True:
            yield function()
    except exception:
        return

class DisplaySubprocessOutputDemo:
    def __init__(self, root):
        self.root = root

        # start dummy subprocess to generate some output
        self.process = Popen([sys.executable, "-u", "-c", dedent("""
            import itertools, time

            for i in itertools.count():
                print("%d.%d" % divmod(i, 10))
                time.sleep(0.1)
            """)], stdout=PIPE)

        # launch thread to read the subprocess output
        #   (put the subprocess output into the queue in a background thread,
        #    get output from the queue in the GUI thread.
        #    Output chain: process.readline -> queue -> label)
        q = Queue(maxsize=1024)  # limit output buffering (may stall subprocess)
        t = Thread(target=self.reader_thread, args=[q])
        t.daemon = True # close pipe if GUI process exits
        t.start()

        # show subprocess' stdout in GUI
        self.label = tk.Label(root, text="  ", font=(None, 200))
        self.label.pack(ipadx=4, padx=4, ipady=4, pady=4, fill='both')
        self.update(q) # start update loop

    def reader_thread(self, q):
        """Read subprocess output and put it into the queue."""
        try:
            with self.process.stdout as pipe:
                for line in iter(pipe.readline, b''):
                    q.put(line)
        finally:
            q.put(None)

    def update(self, q):
        """Update GUI with items from the queue."""
        for line in iter_except(q.get_nowait, Empty): # display all content
            if line is None:
                self.quit()
                return
            else:
                self.label['text'] = line # update GUI
                break # display no more than one line per 40 milliseconds
        self.root.after(40, self.update, q) # schedule next update

    def quit(self):
        self.process.kill() # exit subprocess if GUI is closed (zombie!)
        self.root.destroy()


root = tk.Tk()
app = DisplaySubprocessOutputDemo(root)
root.protocol("WM_DELETE_WINDOW", app.quit)
# center window
root.eval('tk::PlaceWindow %s center' % root.winfo_pathname(root.winfo_id()))
root.mainloop()

Суть решения:

  • поместить вывод подпроцесса в очередь в фоновом потоке
  • получить вывод из очереди в потоке графического интерфейса.

Т.е. вызовите process.readline() в фоновом потоке -> очередь -> обновить метку графического интерфейса в основном потоке. Связано kill-process.py (без опроса - менее переносимое решение, использующее event_generate в фоновом потоке).

4 голосов
/ 20 марта 2009

Перенаправление stdout на метод write (), который обновляет ваш графический интерфейс, является одним из способов, и, вероятно, самым быстрым - хотя запуск подпроцесса, вероятно, является более элегантным решением.

Перенаправляйте stderr только тогда, когда вы действительно уверены, что он работает и работает!

Пример имплиментации (графический файл и тестовый скрипт):

test_gui.py:

from Tkinter import *
import sys
sys.path.append("/path/to/script/file/directory/")

class App(Frame):
    def run_script(self):
        sys.stdout = self
        ## sys.stderr = self
        try:
            del(sys.modules["test_script"])
        except:
            ## Yeah, it's a real ugly solution...
            pass
        import test_script
        test_script.HelloWorld()
        sys.stdout = sys.__stdout__
        ## sys.stderr = __stderr__

    def build_widgets(self):
        self.text1 = Text(self)
        self.text1.pack(side=TOP)
        self.button = Button(self)
        self.button["text"] = "Trigger script"
        self.button["command"] = self.run_script
        self.button.pack(side=TOP)

    def write(self, txt):
        self.text1.insert(INSERT, txt)

    def __init__(self, master=None):
        Frame.__init__(self, master)
        self.pack()
        self.build_widgets()

root = Tk()
app = App(master = root)
app.mainloop()

test_script.py:

print "Hello world!"

def HelloWorld():
    print "HelloWorldFromDef!"
3 голосов
/ 10 апреля 2018

Извините за мой плохой английский. На самом деле я использовал другой способ печати вывода командной строки в моем новом инструменте автоматизации. Пожалуйста, найдите эти шаги ниже.

1> Создать Bat-файл и перенаправить его вывод в LOG-файл. Команда командной строки: tasklist /svc

2> Заставить прочитать этот файл с помощью Python 3.x. `processingFile = open ('D: \ LOG \ taskLog.txt', 'r')

3> Финальный шаг. ttk.Label(Tab4, text=[ProcessFile.read()]).place(x=0, y=27)

** Поэтому, пожалуйста, сообщите, что я еще не включил полосу прокрутки в этот код.

Скриншот публикации:

enter image description here

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...