
标题: [问题求助] 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#代码- dontSaveButton = modalWindows[0].FindFirstDescendant(cf => cf.ByAutomationId("CommandButton_7")).AsButton();
复制代码
所有ai都认为下面这样改, 但是实际使用报错- $dontSaveButton = $modalWindows[0].FindFirstDescendant([FlaUI.Core.Definitions.Condition]::ByAutomationId("CommandButton_7")).AsButton()
复制代码
powershell一向是简化c#代码, 在这里却不行了, 即便写通了, 估计也得一堆代码吧
应该是把相关的代码写在花括号中, 类似下面这样, 但是还是报错- $dontSaveButton = $modalWindows[0].FindFirstDescendant( { param($cf); $cf.ByAutomationId("CommandButton_7") } )
复制代码
作者: 小白龙 时间: 2025-3-21 12:49
C#下面都可以执行, 但是改成powershell死活就是不行- var conditionFactory = window.Automation.ConditionFactory;
- var condition = conditionFactory.ByAutomationId("CommandButton_7");
- dontSaveButton = modalWindows[0].FindFirstDescendant(condition).AsButton();
复制代码
作者: Five66 时间: 2025-3-21 22:13
试试将块转成action或func委托 ,例如- [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, 得先把上游问题解决掉- var modalWindows = Retry.WhileEmpty(() => window.ModalWindows, TimeSpan.FromSeconds(1)).Result;
复制代码
下面代码是gpt生成的powershell, 改的不正确- function CloseWindowWithDontSave {
- param($window)
-
- $window.Close()
- [Wait]::UntilInputIsProcessed()
-
- # 获取窗口的模态对话框
- $modalWindows = [Retry]::WhileEmpty({$window.ModalWindows}, [TimeSpan]::FromSeconds(5)).Result
-
- # 查找 "不保存" 按钮
- if ([Tools.OperatingSystem]::IsWindows11()) {
- $dontSaveButton = $modalWindows[0].FindFirstDescendant("SecondaryButton").AsButton()
- } else {
- $dontSaveButton = $modalWindows[0].FindFirstDescendant( { param($cf) $cf.ByAutomationId("CommandButton_7")} ).AsButton()
- }
-
- # 点击 "不保存" 按钮
- $dontSaveButton.Invoke()
- }
复制代码
原C#代码如下:- public static void CloseWindowWithDontSave(Window window)
- {
- window.Close();
- Wait.UntilInputIsProcessed();
- var modalWindows = Retry.WhileEmpty(() => window.ModalWindows, TimeSpan.FromSeconds(1)).Result;
- Button dontSaveButton;
- if (Tools.OperatingSystem.IsWindows11())
- {
- dontSaveButton = modalWindows[0].FindFirstDescendant("SecondaryButton").AsButton();
- }
- else
- {
- dontSaveButton = modalWindows[0].FindFirstDescendant(cf => cf.ByAutomationId("CommandButton_7")).AsButton();
- }
- dontSaveButton.Invoke();
- }
复制代码
作者: Five66 时间: 2025-3-22 19:44
回复 9# 小白龙
试试- function CloseWindowWithDontSave {
- param($window)
-
- $window.Close()
-
- [FlaUI.Core.Input.Wait]::UntilInputIsProcessed()
-
- # 获取窗口的模态对话框
- $modalWindows = [FlaUI.Core.Tools.Retry]::WhileEmpty([func[[FlaUI.Core.AutomationElements.Window[]]]]{$window.ModalWindows}, [TimeSpan]::FromSeconds(1)).Result
-
- # 查找 "不保存" 按钮
- if ([FlaUI.Core.Tools.OperatingSystem]::IsWindows11()) {
- $dontSaveButton = $modalWindows[0].FindFirstDescendant("SecondaryButton").AsButton()
- } else {
- $dontSaveButton = $modalWindows[0].FindFirstDescendant( { param($cf) $cf.ByAutomationId("CommandButton_7")} ).AsButton()
- }
-
- # 点击 "不保存" 按钮
- $dontSaveButton.Invoke()
- }
复制代码
作者: 小白龙 时间: 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- $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# 小白龙
将- $dontSaveButton = $modalWindows[0].FindFirstDescendant("SecondaryButton").AsButton()
复制代码
和- $dontSaveButton = $modalWindows[0].FindFirstDescendant( { param($cf) $cf.ByAutomationId("CommandButton_7")} ).AsButton()
复制代码
换成- $dontSaveButton=[FlaUI.Core.AutomationElements.AutomationElementExtensions]::AsButton($modalWindows[0].FindFirstDescendant("SecondaryButton"))
复制代码
- $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# 小白龙
参考下面的 ,先打开记事本程序 ,然后往记事本里随便输入一些东西 ,接着运行代码- add-type -LiteralPath ($pwd.path+'\FlaUI.Core.dll')
- add-type -LiteralPath ($pwd.path+'\Interop.UIAutomationClient.dll')
- add-type -LiteralPath ($pwd.path+'\FlaUI.UIA3.dll')
-
- function CloseWindowWithDontSave {
- param($window)
-
- $window.Close()
- [FlaUI.Core.Input.Wait]::UntilInputIsProcessed()
-
- # 获取窗口的模态对话框
- $modalWindows=[FlaUI.Core.Tools.Retry]::WhileEmpty([func[[FlaUI.Core.AutomationElements.Window[]]]]{$window.ModalWindows}, [TimeSpan]::FromSeconds(1),$null,$false,$false,$null).Result
-
- # 查找 "不保存" 按钮
- if ([FlaUI.Core.Tools.OperatingSystem]::IsWindows11()) {
- $dontSaveButton = $modalWindows[0].FindFirstDescendant("SecondaryButton")
- } else {
- $dontSaveButton = $modalWindows[0].FindFirstDescendant( { param($cf) $cf.ByAutomationId("CommandButton_7")} )
- }
-
- if($dontSaveButton){$dontSaveButton=[FlaUI.Core.AutomationElements.AutomationElementExtensions]::AsButton($dontSaveButton)
-
- # 点击 "不保存" 按钮
- $dontSaveButton.Invoke()
- }}
-
- #获取记事本进程
- $ntp=[object[]](ps -Name notepad)
-
- #创建第一个记事本进程Application对象
- $app=[FlaUI.Core.Application]::Attach($ntp[0].id)
-
- #创建UIA3Automation对象
- $auto=[FlaUI.UIA3.UIA3Automation]::new()
-
- #获取记事本进程窗口对象
- $window=$app.GetMainWindow($auto)
-
- #调用函数
- 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 |