Board logo

标题: [问题求助] C#代码中lambda 表达式转成powershell代码 [打印本页]

作者: 小白龙    时间: 2025-3-21 10:48     标题: C#代码中lambda 表达式转成powershell代码

本帖最后由 小白龙 于 2025-3-21 11:40 编辑

我想把下面这行C#代码转为powershell代码, 里面有个lambda 表达式,即cf => cf.ByAutomationId("CommandButton_7")
所有ai把我转晕了, 也没搞定, 求路过大佬支招, 多谢

https://github.com/FlaUI/FlaUI/b ... ilityMethods.cs#L79

原C#代码
  1. dontSaveButton = modalWindows[0].FindFirstDescendant(cf => cf.ByAutomationId("CommandButton_7")).AsButton();
复制代码
所有ai都认为下面这样改, 但是实际使用报错
  1. $dontSaveButton = $modalWindows[0].FindFirstDescendant([FlaUI.Core.Definitions.Condition]::ByAutomationId("CommandButton_7")).AsButton()
复制代码
powershell一向是简化c#代码, 在这里却不行了, 即便写通了, 估计也得一堆代码吧

应该是把相关的代码写在花括号中, 类似下面这样, 但是还是报错
  1. $dontSaveButton = $modalWindows[0].FindFirstDescendant( { param($cf); $cf.ByAutomationId("CommandButton_7") } )
复制代码

作者: 小白龙    时间: 2025-3-21 12:49

C#下面都可以执行, 但是改成powershell死活就是不行
  1. var conditionFactory = window.Automation.ConditionFactory;
  2. var condition = conditionFactory.ByAutomationId("CommandButton_7");
  3. dontSaveButton = modalWindows[0].FindFirstDescendant(condition).AsButton();
复制代码

作者: Five66    时间: 2025-3-21 22:13

试试将块转成action或func委托 ,例如
  1. [func[xxx,yyy]]{$args[0].ByAutomationId("CommandButton_7")}
复制代码
xxx换成参数类型
yyy换成返回类型
作者: 小白龙    时间: 2025-3-21 23:26

回复 3# Five66


    怎么都不行, 真是怪了
作者: jyswjjgdwtdtj    时间: 2025-3-21 23:39

回复 1# 小白龙

param语句后不应当有分号吧
作者: 小白龙    时间: 2025-3-21 23:46

回复 5# jyswjjgdwtdtj


    试了也不行
作者: 小白龙    时间: 2025-3-21 23:54

回复 3# Five66


     发生错误: 方法调用失败,因为 [System.Char] 不包含名为“FindFirstDescendant”的方法。
作者: Five66    时间: 2025-3-22 01:21

回复 7# 小白龙


$modalWindows[0]的问题
就如报错说明的那样[System.Char] 不包含名为“FindFirstDescendant”的方法。也就是 $modalWindows[0] 类型为 [System.Char] ,没有叫做  FindFirstDescendant 的方法
[FlaUI.Core.AutomationElements.AutomationElement] 类型才有名为 FindFirstDescendant 的方法 ,请确保 $modalWindows[0] 的类型为 [FlaUI.Core.AutomationElements.AutomationElement]
作者: 小白龙    时间: 2025-3-22 05:44

本帖最后由 小白龙 于 2025-3-22 05:46 编辑

回复 8# Five66

多谢大佬指导, 确实是哪个问题,
我是想把下面链接的c#函数改为powershell
https://github.com/FlaUI/FlaUI/b ... ilityMethods.cs#L67
现在的问题是, 下面这行C#代码怎样正确改为powershell, 得先把上游问题解决掉
  1. var modalWindows = Retry.WhileEmpty(() => window.ModalWindows, TimeSpan.FromSeconds(1)).Result;
复制代码
下面代码是gpt生成的powershell, 改的不正确
  1. function CloseWindowWithDontSave {
  2.     param($window)
  3.     $window.Close()
  4.     [Wait]::UntilInputIsProcessed()
  5.     # 获取窗口的模态对话框
  6.     $modalWindows = [Retry]::WhileEmpty({$window.ModalWindows}, [TimeSpan]::FromSeconds(5)).Result
  7.     # 查找 "不保存" 按钮
  8.     if ([Tools.OperatingSystem]::IsWindows11()) {
  9.         $dontSaveButton = $modalWindows[0].FindFirstDescendant("SecondaryButton").AsButton()
  10.     } else {
  11.         $dontSaveButton = $modalWindows[0].FindFirstDescendant( { param($cf) $cf.ByAutomationId("CommandButton_7")} ).AsButton()
  12.     }
  13.     # 点击 "不保存" 按钮
  14.     $dontSaveButton.Invoke()
  15. }
复制代码
原C#代码如下:
  1.         public static void CloseWindowWithDontSave(Window window)
  2.         {
  3.             window.Close();
  4.             Wait.UntilInputIsProcessed();
  5.             var modalWindows = Retry.WhileEmpty(() => window.ModalWindows, TimeSpan.FromSeconds(1)).Result;
  6.             Button dontSaveButton;
  7.             if (Tools.OperatingSystem.IsWindows11())
  8.             {
  9.                 dontSaveButton = modalWindows[0].FindFirstDescendant("SecondaryButton").AsButton();
  10.             }
  11.             else
  12.             {
  13.                 dontSaveButton = modalWindows[0].FindFirstDescendant(cf => cf.ByAutomationId("CommandButton_7")).AsButton();
  14.             }
  15.             dontSaveButton.Invoke();
  16.         }
复制代码

作者: Five66    时间: 2025-3-22 19:44

回复 9# 小白龙


试试
  1. function CloseWindowWithDontSave {
  2.     param($window)
  3.     $window.Close()
  4.     [FlaUI.Core.Input.Wait]::UntilInputIsProcessed()
  5.     # 获取窗口的模态对话框
  6.     $modalWindows = [FlaUI.Core.Tools.Retry]::WhileEmpty([func[[FlaUI.Core.AutomationElements.Window[]]]]{$window.ModalWindows}, [TimeSpan]::FromSeconds(1)).Result
  7.     # 查找 "不保存" 按钮
  8.     if ([FlaUI.Core.Tools.OperatingSystem]::IsWindows11()) {
  9.         $dontSaveButton = $modalWindows[0].FindFirstDescendant("SecondaryButton").AsButton()
  10.     } else {
  11.         $dontSaveButton = $modalWindows[0].FindFirstDescendant( { param($cf) $cf.ByAutomationId("CommandButton_7")} ).AsButton()
  12.     }
  13.     # 点击 "不保存" 按钮
  14.     $dontSaveButton.Invoke()
  15. }
复制代码

作者: 小白龙    时间: 2025-3-23 00:00

回复 10# Five66


    不行报错

C:\Users\Administrator\Desktop\FLAUI\_MouseTests_Click.ps1 : 发生错误: 找不到“WhileEmpty”的重载,参数计数为:“2”。
    + CategoryInfo          : NotSpecified: ( [Write-Error], WriteErrorException
    + FullyQualifiedErrorId : Microsoft.PowerShell.Commands.WriteErrorException,_MouseTests_Click.ps1
作者: 小白龙    时间: 2025-3-23 00:04

本帖最后由 小白龙 于 2025-3-23 01:04 编辑

回复 10# Five66


    大佬, 我在把下面链接中的C#函数改为powershell, 上面要转的代码就是, 试两天了, 都没结果
https://github.com/FlaUI/FlaUI/b ... s/MouseTests.cs#L27
作者: Five66    时间: 2025-3-23 01:06

回复 11# 小白龙


    试试出错那行换成 ,不行的话估计得调用泛型版本的WhileEmpty方法 ,ps7.3之前调用net泛型方法挺麻烦的 ,自己网上搜索 ,或者干脆试试直接换成 $modalWindows=$window.ModalWindows
  1. $modalWindows = [FlaUI.Core.Tools.Retry]::WhileEmpty([func[[FlaUI.Core.AutomationElements.Window[]]]]{$window.ModalWindows}, [TimeSpan]::FromSeconds(1),$null,$false,$false,$null).Result
复制代码

作者: 小白龙    时间: 2025-3-23 01:14

回复 13# Five66


    还是不行
C:\Users\Administrator\Desktop\FLAUI\_MouseTests_Click.ps1 : 发生错误: 方法调用失败,因为 [FlaUI.Core.AutomationElements.AutomationElement] 不包含名为“AsButton”的
方法。
    + CategoryInfo          : NotSpecified: ( [Write-Error], WriteErrorException
    + FullyQualifiedErrorId : Microsoft.PowerShell.Commands.WriteErrorException,_MouseTests_Click.ps1
作者: Five66    时间: 2025-3-23 02:02

回复 14# 小白龙


    将
  1. $dontSaveButton = $modalWindows[0].FindFirstDescendant("SecondaryButton").AsButton()
复制代码
  1. $dontSaveButton = $modalWindows[0].FindFirstDescendant( { param($cf) $cf.ByAutomationId("CommandButton_7")} ).AsButton()
复制代码
换成
  1. $dontSaveButton=[FlaUI.Core.AutomationElements.AutomationElementExtensions]::AsButton($modalWindows[0].FindFirstDescendant("SecondaryButton"))
复制代码
  1. $dontSaveButton=[FlaUI.Core.AutomationElements.AutomationElementExtensions]::AsButton($modalWindows[0].FindFirstDescendant( { param($cf) $cf.ByAutomationId("CommandButton_7")} ))
复制代码

作者: 小白龙    时间: 2025-3-23 06:13

回复 15# Five66


    C:\Users\Administrator\Desktop\FLAUI\_MouseTests_Click.ps1 : 发生错误: 使用“0”个参数调用“Invoke”时发生异常:“The requested pattern 'Invoke [#10000]' is not supp
orted”
    + CategoryInfo          : NotSpecified: ( [Write-Error], WriteErrorException
    + FullyQualifiedErrorId : Microsoft.PowerShell.Commands.WriteErrorException,_MouseTests_Click.ps1
作者: Five66    时间: 2025-3-23 09:12

回复 16# 小白龙


    貌似是这样的 ,错误是由flaui抛出的 ,并不是ps本身的错误
原来c#的是测试代码 ,ps代码的话自己根据实际来 ,可以看看ps代码里的那个$dontSaveButton是什么
作者: 小白龙    时间: 2025-3-23 10:32

回复 17# Five66


    这个就是咱们在关闭记事本的时候, 弹出来的那个, 保存的对话框, 这代码相当于点了 否
作者: Five66    时间: 2025-3-23 23:56

回复 18# 小白龙


    这样的话 ,调用那个 FindFirstDescendant 方法的参数就不是SecondaryButton和CommandButton_7了
看错误提示那个#10000 ,大概是用窗口类名的 ,自己换成记事本对应的窗口类 ,比如#32770是对话框的类名
还有如果不是用ps代码启动的记事本 ,还得找到记事本的窗口然后调用CloseWindowWithDontSave函数 ,总之就自己按照实际来
;
像这种还不直接调winapi ,FindWindow FindWindowEx SendMessage简单
作者: 小白龙    时间: 2025-3-24 15:23

回复 19# Five66


    他这个函数就是一个通用的关闭对话框的函数, 在所有语言的操作系统上都适用, 不管是英文还是日文系统, 所以感觉很有用, 就想转换一下powershell, 大佬能方便的时候再看看吗?
作者: Five66    时间: 2025-3-25 00:50

本帖最后由 Five66 于 2025-3-25 00:54 编辑

回复 20# 小白龙


    参考下面的 ,先打开记事本程序 ,然后往记事本里随便输入一些东西 ,接着运行代码
  1. add-type -LiteralPath ($pwd.path+'\FlaUI.Core.dll')
  2. add-type -LiteralPath ($pwd.path+'\Interop.UIAutomationClient.dll')
  3. add-type -LiteralPath ($pwd.path+'\FlaUI.UIA3.dll')
  4. function CloseWindowWithDontSave {
  5.    param($window)
  6.    $window.Close()
  7.    [FlaUI.Core.Input.Wait]::UntilInputIsProcessed()
  8.     # 获取窗口的模态对话框
  9.     $modalWindows=[FlaUI.Core.Tools.Retry]::WhileEmpty([func[[FlaUI.Core.AutomationElements.Window[]]]]{$window.ModalWindows}, [TimeSpan]::FromSeconds(1),$null,$false,$false,$null).Result
  10.     # 查找 "不保存" 按钮
  11.     if ([FlaUI.Core.Tools.OperatingSystem]::IsWindows11()) {
  12.         $dontSaveButton = $modalWindows[0].FindFirstDescendant("SecondaryButton")
  13.     } else {
  14.         $dontSaveButton = $modalWindows[0].FindFirstDescendant( { param($cf) $cf.ByAutomationId("CommandButton_7")} )
  15.     }
  16. if($dontSaveButton){$dontSaveButton=[FlaUI.Core.AutomationElements.AutomationElementExtensions]::AsButton($dontSaveButton)
  17.     # 点击 "不保存" 按钮
  18.     $dontSaveButton.Invoke()
  19. }}
  20. #获取记事本进程
  21. $ntp=[object[]](ps -Name notepad)
  22. #创建第一个记事本进程Application对象
  23. $app=[FlaUI.Core.Application]::Attach($ntp[0].id)
  24. #创建UIA3Automation对象
  25. $auto=[FlaUI.UIA3.UIA3Automation]::new()
  26. #获取记事本进程窗口对象
  27. $window=$app.GetMainWindow($auto)
  28. #调用函数
  29. CloseWindowWithDontSave($window)
复制代码

作者: Five66    时间: 2025-3-25 01:07

回复 21# Five66


    补充点 ,上面说的记事本是指系统自带的记事本 ,其他记事本的话可能因为窗口层次关系或窗口类名不同而出错 ,反正就是自己按实际来
作者: 小白龙    时间: 2025-3-25 03:30

本帖最后由 小白龙 于 2025-3-25 03:31 编辑

回复 21# Five66


    多谢大佬, 这回行了, 看了一下, 原来是这行代码不行
$modalWindows = [Retry]::WhileEmpty({$window.ModalWindows}, [TimeSpan]::FromSeconds(1)).Result
改成下面就行了,
$modalWindows = [FlaUI.Core.Tools.Retry]::WhileEmpty([func[[FlaUI.Core.AutomationElements.Window[]]]]{$window.ModalWindows}, [TimeSpan]::FromSeconds(1),$null,$false,$false,$null).Result
原C#是
var modalWindows = Retry.WhileEmpty(() => window.ModalWindows, TimeSpan.FromSeconds(1)).Result;
为什么多了好几个参数呢
作者: 小白龙    时间: 2025-3-25 03:39

回复 21# Five66


    我看cs的定义, 如下图, 其它几个参数都有默认值的, C#只用了两个参数就行了, 为啥powershell要补全呢

作者: 小白龙    时间: 2025-3-25 03:52

回复 21# Five66


    问了一下GPT, 说可以省啊!, 下面是gpt的解释
如果 C# 方法 **没有重载**,但有 **可选参数**,在 PowerShell 中仍然 **可以省略这些可选参数**,方法会使用默认值。  

---

## **示例:C# 只有一个方法(无重载),但有可选参数**
假设 C# 代码如下:
```csharp
public class MyClass
{
    public void MyMethod(int a, string b = "defaultB", int c = 10, bool d = false)
    {
        Console.WriteLine($"a={a}, b={b}, c={c}, d={d}");
    }
}
```

在 PowerShell 中调用:
```powershell
# 创建对象
$object = New-Object MyClass

# 只传递必需参数 `a`,省略 `b`, `c`, `d`
$object.MyMethod(1)
# 输出: a=1, b=defaultB, c=10, d=False

# 传递 `a`, `b`,省略 `c`, `d`
$object.MyMethod(1, "CustomB")
# 输出: a=1, b=CustomB, c=10, d=False

# 传递 `a`, `b`, `c`,省略 `d`
$object.MyMethod(1, "CustomB", 20)
# 输出: a=1, b=CustomB, c=20, d=False

# 传递所有参数
$object.MyMethod(1, "CustomB", 20, $true)
# 输出: a=1, b=CustomB, c=20, d=True
```

---

## **PowerShell 省略可选参数的规则**
1. **PowerShell 会自动使用 C# 方法定义的默认值**。
2. **参数必须按顺序传递**,不能直接跳过中间参数。例如:
   ```powershell
   $object.MyMethod(1, , , $true)  # ❌ 错误,PowerShell 不允许省略中间的逗号
   ```
   如果需要指定后面的参数,必须传 `null`:
   ```powershell
   $object.MyMethod(1, $null, $null, $true)  # ✅ 这样可以
   ```

---

## **结论**
✅ **C# 只有一个方法,没有重载,但有可选参数时,PowerShell 仍然可以省略可选参数**。  
✅ **必须按顺序传递参数,不能直接跳过中间参数**,如果想跳过,可以传 `$null` 作为占位符。  

你在转换 C# 代码时遇到了什么问题?可以贴出来,我可以帮你看看!😊
作者: Five66    时间: 2025-3-25 07:46

回复 23# 小白龙


    啊 ,WhileEmpty方法调用代码13楼不是发过了
需要调用泛型的WhileEmpty方法 ,C#有编译支持 ,可以指定泛型方法的类型 ,也有参数演绎
但是ps并没有参数演绎 ,ps版本7.3才能指定泛型方法的类型 ,因此ps版本7.3以下调用泛型方法要么依靠ps本身的参数推断 ,要么使用反射 ,省略参数不行就带上全部参数 ,带上参数不行就用反射 ,就是这样的规则和设定
ps调用泛型方法可参考:https://learn.microsoft.com/zh-c ... view=powershell-7.3
作者: 小白龙    时间: 2025-3-25 08:14

回复 22# Five66
大佬像下面这种的, 都得像上面那样改吗? 怎么个更改思路? 对C#还不熟悉

            var mainWindow = Application.GetMainWindow(Automation);
            var tab = mainWindow.FindFirstDescendant(cf => cf.ByControlType(ControlType.Tab)).AsTab();
            tab.SelectTabItem(1);
            var grid = mainWindow.FindFirstDescendant(cf => cf.ByAutomationId("listView1")).AsGrid();
作者: Five66    时间: 2025-3-25 08:54

回复 27# 小白龙


   
普通方法直接改
拓展方法改成对应的静态方法调用 ,AsTab ,AsButton ,AsCalendar ,Asxxx....之类的都是FlaUI.Core.AutomationElements.AutomationElementExtensions类的静态方法
泛型方法改的时候尽量指定参数类型并带上所有参数 ,不行就只能反射 ,反射参考26楼的链接
lambda改成块或func或action
不知道啥类型的在ps里gm一下或者直接看源码
作者: 小白龙    时间: 2025-3-25 09:41

回复 28# Five66

多谢大佬指导, 能帮看一下这个问题吗? 多谢
http://www.bathome.net/thread-70714-1-1.html




欢迎光临 批处理之家 (http://www.bathome.net/) Powered by Discuz! 7.2