我正在尝试在我的应用程序中使用MS UI 自动化框架进行一些自动化测试。我现在的目标是监听 GUI 事件并记录它们,类似于 Windows SDK 中提供的示例 (TestScriptGeneratorSample)。
只需使用该演示应用程序,我就可以使用一个按钮启动一个简单的 Windows 窗体应用程序,并看到它将单击事件记录为“调用”UIA 事件。但是,对于按钮的每次单击,演示应用程序都会记录4 个“调用”事件。这是为什么?使用 WPF 按钮时,我没有看到这个问题。
我的猜测是,由于System.Windows.Forms.Button
该类支持MSAA而不是 UIA,因此桥接 MSAA 接口的 UIA 部分行为不端,或者至少以我不理解且找不到任何文档的方式表现。也许它正在报告鼠标按下,向上,单击,然后实际按下按钮的调用事件?
谁能解释这种行为,和/或提供一种解决方法,以便按下一个按钮会导致一个调用事件?
编辑:这是在 WinXP SP3 下。我刚刚安装了Windows 自动化 API 3.0 更新,仍然看到相同的行为。
编辑2:我发现了这个例子,作者不经意地提到了这种行为作为Win32控件中的一个错误,但没有引用任何证据......
这是我的示例应用程序(上面有一个按钮的表单),以及其中包含的事件监听。添加对UIAutomationClient
和的引用UIAutomationTypes
。单击按钮并看到调用发生了四次而不是一次。
using System;
using System.Drawing;
using System.Windows.Automation;
using System.Windows.Forms;
namespace WindowsFormsApplication6
{
public partial class Form1 : Form
{
private TextBox m_Output;
public Form1()
{
InitializeComponent();
// set up the GUI with a single button...
Button b = new Button {Location = new Point(5, 5), Name = "Test", Text = "Click Me"};
// ... and a textbox to write events to.
m_Output = new TextBox { Location = new Point(5, 30), Name = "Output", Multiline = true, Size = new Size(250, 200) };
this.Controls.Add(b);
this.Controls.Add(m_Output);
// get the button as an UIA element
AutomationElement button = AutomationElement.FromHandle(b.Handle);
// attach a handler to the button, listening for the Invoke event
Automation.AddAutomationEventHandler(
InvokePattern.InvokedEvent,
button,
TreeScope.Element,
OnInvoked);
}
// Handler for button invoke events
private void OnInvoked(object Sender, AutomationEventArgs E)
{
AppendText("Invoked: " + ((AutomationElement)Sender).Current.AutomationId);
}
// Just dumps text to the textbox
private void AppendText(string Text)
{
if (m_Output.InvokeRequired)
{
m_Output.BeginInvoke((Action<string>)AppendText, Text);
}
else
{
m_Output.AppendText(DateTime.Now.ToString("hh:mm:ss.fff") + ": " + Text + Environment.NewLine);
}
}
}
}