.net 如何使用UI自动化从ListView或类似控件中获取文本?

mzillmmw  于 2022-12-20  发布在  .NET
关注(0)|答案(1)|浏览(283)

我正在尝试从外部应用程序中抓取一个类似ListView的控件。现在我使用System.Windows.Automation。使用AutoIt v3,我提取了以下关于要从中抓取文本的确切控件的信息:

>>>> Control <<<<
Class:  WindowsForms10.Window.8.app.0.34f5582_r6_ad1
Instance:   20
ClassnameNN:    WindowsForms10.Window.8.app.0.34f5582_r6_ad120
Name:   
Advanced (Class):   [CLASS:WindowsForms10.Window.8.app.0.34f5582_r6_ad1; INSTANCE:20]
ID: 1510520
Text:   
Position:   182, 164
Size:   1411, 639
ControlClick Coords:    300, 202
Style:  0x56010000
ExStyle:    0x00000000
Handle: 0x0000000000170C78

现在,我已经注意到ID = 1510520,通过使用它,我将能够获得控制

AutomationElement element = AutomationElement.FromHandle(1510520);

该控件看起来像一个ListView或类似的,但我不能用它做任何其他事情。

现在如何获取此控件的内容?
更新:
感谢Jimi的推荐,Windows 10 SDK中的inspect.exe运行得最好!我能够深入到DataGridView。

cvxl0en2

cvxl0en21#

我假设您可以找到包含要从中提取数据的DataGridView的Window,GeDataGridViewDataTable()方法需要该Window的Handle。
让我们分解这些方法:
要在已知句柄的情况下获取感兴趣的窗口的AutomationElement,我们可以只使用window =AutomationElement.FromHandle([Window Handle])
▶这里我使用AndCodition,因为你可能有ProcessID和Window Title,所以你可以使用AutomationElement.ProcessIdPropertyAutomationElement.NameProperty作为条件,而不是使用AutomationElement.ControlTypePropertyAutomationElement.NativeWindowHandleProperty进行过滤。
如果找到Window,则解析**TreeScope.SubTree**作用域中的第一个子元素(该Window中的所有UI元素),以找到Table(ControlType.Table)类型的第一个元素。
▶当然,Windows可能托管多个DataGridView:在这种情况下,我们可以使用FindAll()代替FindFirst(),然后使用一些其他条件(列的数量、标题的文本、单元格的内容、位置、大小、父容器等)来确定哪个是哪个。
当找到感兴趣的DataGridView时,我们可以提取其Cells的内容。
下面是第二种方法GetDataGridViewRowsCollection()

  • 第一个OrCondition过滤掉DataGridView滚动条,可能还有网格的其他子控件(定制可能包括一些)。
  • 之后,我们检查DGV是否有标题:如果有,那么第一个子Row元素的名称是**Top Row**。然后,我们可以使用标题文本来命名DataTable的列,该列将存储提取的数据。否则,只需添加一些默认名称。
  • 然后,对于每个Row元素,我们枚举它的子元素,代表Cells。我添加了一个NotCondition过滤器来排除行标题单元格**ControlType.Header**(如果有的话)。
  • 然后,我们迭代Cell的集合,使用GetCurrentPropertyValue()方法提取它们的值,将属性类型设置为ValuePattern.ValueProperty,将这些值添加到一个List中,该List将提供param数组参数**DataTable.Rows.Add()**。
private DataTable GeDataGridViewDataTable(IntPtr windowHwnd)
{
    var condition = new AndCondition(
        new PropertyCondition(AutomationElement.ControlTypeProperty, ControlType.Window),
        new PropertyCondition(AutomationElement.NativeWindowHandleProperty, windowHwnd.ToInt32())
    );
    var window = AutomationElement.RootElement.FindFirst(TreeScope.Children, condition);
    if (window == null) return null;
    var dgv = window.FindFirst(TreeScope.Subtree, 
        new PropertyCondition(AutomationElement.ControlTypeProperty, ControlType.Table));

    if (dgv == null) return null;
    var dt = GetDataGridViewRowsCollection(dgv);
    return dt;
}

private DataTable GetDataGridViewRowsCollection(AutomationElement dgv)
{
    var dt = new DataTable();

    // Skips ScrollBars and other child elements
    var condition = new OrCondition(
        new PropertyCondition(AutomationElement.ControlTypeProperty, ControlType.Custom),
        new PropertyCondition(AutomationElement.ControlTypeProperty, ControlType.Header)
    );

    var rows = dgv.FindAll(TreeScope.Children, condition).OfType<AutomationElement>().ToList();
    bool hasColumnHeader = (rows[0].Current.Name == "Top Row");

    // First element is the Header (if there's one)
    var dgvHeaderColumns = rows[0].FindAll(TreeScope.Children, Condition.TrueCondition);
        
    // Skip the Top/Left header
    for (int i = 1; i < dgvHeaderColumns.Count; i++) {
        dt.Columns.Add(hasColumnHeader ? dgvHeaderColumns[i].Current.Name : "Column"+i);
    }

    // Skips the Row Header, if any
    var notCondition = new NotCondition(new PropertyCondition(AutomationElement.ControlTypeProperty, ControlType.Header));
    foreach (AutomationElement row in rows) {
        var cells = row.FindAll(TreeScope.Children, notCondition);
        var values = new List<object>();
        foreach (AutomationElement cell in cells) {
            values.Add(cell.GetCurrentPropertyValue(ValuePattern.ValueProperty));
        }
        dt.Rows.Add(values.ToArray());
    }
    return dt;
}

相关问题