Python Tkinter GUI 自动化

Posted

技术标签:

【中文标题】Python Tkinter GUI 自动化【英文标题】:Python Tkinter GUI Automation 【发布时间】:2020-10-01 08:47:48 【问题描述】:

我想进入 GUI 自动化,以便在我自己的程序上运行测试。我要测试的程序是用 Python 编写的,并使用 Tkinter 作为 GUI。测试代码虽然不一定要在 python 中,但 CPP 也可以。我做了一些研究,我已经面临一个问题。

根据我的研究,我发现“Windows 应用程序驱动程序”是一种测试 GUI 的免费方法。还有“WinAppDriver UI Recorder”,使用起来很方便。此外,(在我的情况下)“C:\Program Files (x86)\Windows Kits\10\bin\10.0.19041.0\x86”中的“Inspect.exe”程序对于获取有关 GUI 元素的信息很有用。

假设我有一个像这样的小 Python 代码(仅用于测试):

from Tkinter import *
import ttk

class Test():
    def __init__(self):
        self.root = Tk()
        self.root.geometry("250x100")
        self.text = StringVar()
        self.text.set("Original Text")
        self.buttonA = Button(self.root, textvariable=self.text)
        self.buttonA.configure(text="test")

        self.buttonB = Button(self.root,
                                text="Click to change text",
                                command=self.changeText
                              )
        self.buttonA.pack(side=LEFT)
        self.buttonB.pack(side=RIGHT)
        self.root.mainloop()

    def changeText(self):
        self.text.set("Updated Text")

app=Test()

当运行代码并使用Inspect.exe 检查buttonB 时,我得到的名称是“”(空)。有什么方法可以将该名称更改为具有信息性和有用性的名称,例如计算器示例中的“7”按钮的名称为“Seven”。然后像这样在测试器中使用它:

self.driver.find_element_by_name("Seven").click()

应该是这样的:

self.driver.find_element_by_name("buttonB").click()

以我为例。

【问题讨论】:

【参考方案1】:

您可以将 tkinter 小部件命名为:

self.buttonA = Button(self.root, textvariable=self.text,name = 'buttonA')

如果WinAppDriver 无法找到以这种方式命名的 tkinter 小部件。您可以修改代码以模仿 UI 自动化框架的方式调用按钮(并“保持”其他 UI 小部件):

我修改了您的示例以展示如何做到这一点

from Tkinter import *
import ttk

def _widgets_by_name(parent,name,widgets):
    if not parent.winfo_children():
        if name == parent.winfo_name() :
            widgets.append(parent)
    else:
        for child in parent.winfo_children():
            _widgets_by_name(child,name,widgets)
            
def find_widget_by_name(parent,name):
    ''' ui automation function that can find a widget in an application/hierarchy of widgets by its name '''
    widgets = []
    _widgets_by_name(parent,name,widgets)
    if len(widgets) == 0:
        raise Exception(f'no widget named name found')
    elif len(widgets) >1:
        raise Exception(f'multiple widget named name found')
    return (widgets[0])


class Test():
    def __init__(self):
        self.root = Tk()
        self.root.geometry("250x100")
        self.text = StringVar()
        self.text.set("Original Text")
        self.buttonA = Button(self.root, textvariable=self.text,name = 'button-a')
        self.buttonA.configure(text="test")

        self.buttonB = Button(self.root,
                                text="Click to change text",
                                command=self.changeText,
                                name = 'button-b'
                              )
        self.buttonA.pack(side=LEFT)
        self.buttonB.pack(side=RIGHT)
        # self.root.mainloop() do not start the main loop for testing purpose
        # can still be started outside of  __init__ for normal operation 

    def changeText(self):
        self.text.set("Updated Text")
    
app=Test()

# test the app step by step
# find one of the buttons and invoke it
find_widget_by_name(app.root,'button-b').invoke()
app.root.update() # replace the app mainloop: run the UI refresh once.

assert app.text.get() == "Updated Text"

【讨论】:

我尝试了几乎所有我在互联网上可以找到的与命名 tkinter 小部件有关的东西,但对在 WinAppDriver 中识别它们没有帮助。

以上是关于Python Tkinter GUI 自动化的主要内容,如果未能解决你的问题,请参考以下文章

Tkinter 8.5 GUI for Python-----01

如何使用 Tkinter 创建自动更新的 GUI?

如何让tkinter自动关闭窗口后显示另一个窗口?

python tkinter gui库 问题 gif的图片不能显示 高手帮我看看出错在哪里?

python学习tkinter(1-3)

python GUI(Tkinter)