Тестирование приложения tkinter - PullRequest
0 голосов
/ 23 мая 2018

Я написал небольшое приложение, используя python 3 и tkinter.Тестирование каждого виджета, хотя их не так много, кажется утомительным, поэтому я хотел написать пару автоматических тестов, чтобы упростить процесс.Я прочитал какой-то другой вопрос, который, казалось, имел отношение к этой проблеме, но ни один не соответствовал моим потребностям.Прямо сейчас я делаю тестирование очень простым способом - я вызываю команду для каждого виджета и вручную нажимаю на него, чтобы посмотреть, работает ли он.Это делает вещи немного быстрее, но я постоянно сталкиваюсь с некоторыми проблемами - то есть я не могу автоматически закрывать всплывающие окна (например, showinfo), даже используя библиотеки для имитации нажатий клавиш (а именно pynput).Есть ли эффективный подход для тестирования приложений с использованием tkinter?

Вот код, который я использую сейчас:

import tkinter as tkinter
import unittest
from mygui import MyGUI

class TKinterTestCase(unittest.TestCase):
    def setUp(self):
        self.root = tkinter.Tk()


    def tearDown(self):
        if self.root:
            self.root.destroy()

    def test_enter(self):
        v = MyGUI(self.root)
        v.info_button.invoke()
        v.close_button.invoke()
        v.btnOut.invoke()


if __name__ == "__main__":
    unittest.main()

1 Ответ

0 голосов
/ 23 мая 2018

Я не знаю много о unittest, но я нашел обходной путь для закрытия всплывающих диалогов, таких как showinfo, во время тестов.Идея состоит в том, чтобы использовать событие клавиатуры, чтобы вызвать кнопку диалога.Но так как приложение ожидает, пока пользователь закроет всплывающее диалоговое окно, нам нужно заранее запланировать событие клавиатуры, используя after:

self.root.after(100, self.root.event_generate('<Return>'))
v.button.invoke()

Полный пример

import tkinter
from tkinter import messagebox
import unittest


class MyGUI(tkinter.Frame):
    def __init__(self, master, **kw):
        tkinter.Frame.__init__(self, master, **kw)
        self.info_button = tkinter.Button(self, command=self.info_cmd, text='Info')
        self.info_button.pack()
        self.quit_button = tkinter.Button(self, command=self.quit_cmd, text='Quit')
        self.quit_button.pack()

    def info_cmd(self):
        messagebox.showinfo('Info', master=self)

    def quit_cmd(self):
        confirm = messagebox.askokcancel('Quit?', master=self)
        if confirm:
            self.destroy()


class TKinterTestCase(unittest.TestCase):
    def setUp(self):
        self.root = tkinter.Tk()
        self.root.bind('<Key>', lambda e: print(self.root, e.keysym))

    def tearDown(self):
        if self.root:
            self.root.destroy()

    def test_enter(self):
        v = MyGUI(self.root)
        v.pack()
        self.root.update_idletasks()

        # info
        v.after(100, lambda: self.root.event_generate('<Return>'))
        v.info_button.invoke()

        # quit
        def cancel():
            self.root.event_generate('<Tab>')
            self.root.event_generate('<Return>')

        v.after(100, cancel)
        v.quit_button.invoke()
        self.assertTrue(v.winfo_ismapped())    
        v.after(100, lambda: self.root.event_generate('<Return>'))
        v.quit_button.invoke()
        with self.assertRaises(tkinter.TclError):
            v.winfo_ismapped()


if __name__ == "__main__":
    unittest.main()
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...