在PowerShell中正确显示托盘气泡提示

8

简短版: 我认为我需要帮助正确使用 PowerShell 中的事件,以便通过 Windows 消息触发来消除气球提示的图标。

详细版:

我有一个长时间运行的 PowerShell 命令(构建),我希望在系统托盘/通知区域通过气球提示收到它完成的通知。

我能够创建一个 Write-BalloonTip 脚本(如下)来实现大致所需的功能。唯一的问题是,就像托盘图标经常发生的情况一样,托盘图标直到我将鼠标悬停在其上才会消失。通过重用同一全局变量来表示 NotifyIcon,我可以重复使用此脚本,并保持只剩下一个系统托盘图标(直到我将鼠标悬停在其上)。这仍然感觉像是一个 hack。我尝试添加事件处理程序,以便在 BalloonTipClosed 事件上得到通知,然后在那里处理它。在事件处理程序中,我尝试了三种我看到的建议用于消除残留图标的技术,但都无济于事。

令人恼火的是,简单的 .Dispose 似乎在脚本的后续调用中起作用,这让我认为事件脚本块根本没有被调用。

我已经验证 BalloonTipClosed 在一个单独的 WinForms 应用程序中在提示消失后被调用。

我是否遗漏了一些基本的东西?非常感谢任何帮助。谢谢!

这是 "Write-BalloonTip.ps1" 的代码:

param
(
    $text,
    $title = "",
    $icon = "Info",
    $timeout=15000
)

[System.Reflection.Assembly]::LoadWithPartialName("System.Windows.Forms") | out-null
[System.Reflection.Assembly]::LoadWithPartialName("System.Drawing") | out-null

if ($global:writeBalloonTipIcon)
{
    # This gets rid of the previous one
    $global:writeBalloonTipIcon.Dispose()
}

$global:writeBalloonTipIcon = new-object System.Windows.Forms.NotifyIcon 
$global:writeBalloonTipIcon.Icon = [System.Drawing.SystemIcons]::Information

# FIXME: This *should* cleanup the icon after it's done, but it doesn't seem to work
$global:writeBalloonTipIcon.add_BalloonTipClosed(
  {
    # this *should* work, but it's not. What am I missing?
    $global:writeBalloonTipIcon.Icon = $null; 
    $global:writeBalloonTipIcon.Visible = $false; 
    $global:writeBalloonTipIcon.Dispose();
  });

$global:writeBalloonTipIcon.Visible = $true;
$global:writeBalloonTipIcon.ShowBalloonTip($timeout, $title, $text, $icon);

也许有点离题,但我发现Growl for Windows非常有用。Jaykul编写了一个库来包装这个功能。还有一个插件可以在构建/重建完成时通知您。您可以在此处找到它:http://huddledmasses.org/more-growl-for-windows-from-powershell/(我不知道它是否是最新版本) - stej
并不是真的,没有快速解决方案,所以我们只是一直回收一个全局图标。这是一个合理的解决方法,直到我们通过处理STA问题而有动力真正解决它。 - Jeff Moser
2个回答

2

我认为您需要在STA线程中执行此代码。PowerShell(此处显示v2)默认情况下在MTA线程中执行:

PS U:\> [System.Threading.Thread]::CurrentThread


ManagedThreadId    : 5
ExecutionContext   : System.Threading.ExecutionContext
Priority           : Normal
IsAlive            : True
IsThreadPoolThread : False
IsBackground       : False
ThreadState        : Running
ApartmentState     : MTA
CurrentUICulture   : en-US
CurrentCulture     : en-US
Name               : Pipeline Execution Thread

我认为这更接近问题的核心,因为通知事件似乎是通过Windows消息发送的。有什么正确的方法吗?似乎Invoke-Apartment应该可以工作,但我无法完全让它工作。 - Jeff Moser

1
我建议使用Register-ObjectEvent来订阅BalloonTipClosed事件。这在最近的另一个SO帖子中提到过。看一下吧。

1
我认为问题的核心是STA问题,因为除非在STA线程上,否则事件不会被触发。 - Jeff Moser
JIT调试器报告“System.Management.Automation.PSInvalidOperationException:此线程中没有可用于运行脚本的Runspace。您可以在System.Management.Automation.Runspaces.Runspace类型的DefaultRunspace属性中提供一个。”在BallonTipClosed事件上,因此事件正在被触发。您可以在PS中创建一个Runspace,但使用Register-ObjectEvent要快得多。 - Monso

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接