如何在企业设置中将通知发送到用户/机器



当前情况:我创建了一个PowerShell脚本,可以在屏幕的右下侧绘制类似烤面包的通知。我唯一的问题是,这仅在本地计算机上起作用。如何使此通知显示在目标机器上?仅在Active Directory中指定机器名称?

当然,我对其他建议持开放态度。我目前正在阅读SignalR,但我不确定如何使SignalR在本机台式机上工作,而不是通过网站上的工作。完美的情况将是一个通知系统,可以大规模发送Windows 10中的敬酒通知。

有什么想法?

这是我当前的脚本

Function ShowNotification
{
    [CmdletBinding()]
    param(
           [string] $Title,
           [string] $Message,
           [string] $Image,
           [string] $Hyperlink 
    )
    Add-Type -AssemblyName presentationframework, System.Windows.Forms
         $screenHeight = Get-WmiObject -Class Win32_DesktopMonitor | Select-Object ScreenHeight
         $screenWidth = Get-WmiObject -Class Win32_DesktopMonitor | Select-Object ScreenWidth
         ############################ NOTIFICATION CLIENT GUI #########################################
$XAML2 = @'
<Window Name="Form2"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:ed="http://schemas.microsoft.com/expression/2010/drawing" 
        Title="Notification" Height="141.071" Width="504.611" WindowStyle="None" ShowInTaskbar="False" ResizeMode="NoResize" Background="#313130">
    <Window.Effect>
        <DropShadowEffect/>
    </Window.Effect>
    <Grid Margin="-99,133.5,-102,54" Background="#313130">
        <Grid.ColumnDefinitions>
            <ColumnDefinition Width="238*"/>
            <ColumnDefinition Width="10*"/>
            <ColumnDefinition Width="9*"/>
            <ColumnDefinition Width="190*"/>
            <ColumnDefinition Width="72*"/>
            <ColumnDefinition Width="55*"/>
            <ColumnDefinition Width="132*"/>
        </Grid.ColumnDefinitions>
        <Label Name="TitleLabel" Content="HEY" HorizontalAlignment="Left" Margin="110,-127,0,0" VerticalAlignment="Top" Height="35" Width="388" Grid.ColumnSpan="5" Foreground="White" Background="{x:Null}" FontWeight="Regular" FontSize="18"  FontFamily="Segoe UI SemiLight"/>
        <Path Data="M99,-92.5" Fill="#FFF4F4F5" HorizontalAlignment="Left" Height="1" Margin="99,-92.5,0,0" Stretch="Fill" Stroke="Black" VerticalAlignment="Top" Width="1"/>
        <Image Name="imageBox" Source="<PictureSource>" Grid.Column="5" HorizontalAlignment="Left" Height="120" Margin="0,-114,0,-6" VerticalAlignment="Top" Width="75" Grid.ColumnSpan="2"/>
        <TextBlock Name="MessageTB" HorizontalAlignment="Left" Margin="116,-87,0,0" TextWrapping="Wrap"  VerticalAlignment="Top" Grid.ColumnSpan="5" Height="80" Width="382" Foreground="White" FontFamily="Segoe UI Light" FontSize="14"/>
    </Grid>
</Window>

'@
[xml]$XAML2 = $XAML2 -replace "<PictureSource>", $Image
        ###############################################################################################
        ############################# CONVERT GUI COMPONENTS TO VARIABLES #############################
        $script:window = [Windows.Markup.XamlReader]::Load((New-Object System.Xml.XmlNodeReader $xaml2))
        $xaml2.SelectNodes("//*[@Name]") | ForEach-Object { Set-Variable -Name ($_.Name) -Value $window.FindName($_.Name) -Scope Script }
        ###############################################################################################
        ############################# EVENT HANDLERS ##################################################
        $window.Add_MouseDown({
            If($Image)
            {
                $img = [System.Drawing.Image]::Fromfile($Image);
                $form = new-object Windows.Forms.Form
                $form.Text = "Image Viewer"
                $form.Width = $img.Size.Height;
                $form.Height =  $img.Size.Width;
                $form.StartPosition = "CenterScreen"
                $pictureBox = new-object Windows.Forms.PictureBox
                $pictureBox.Dock = "Fill"
                $pictureBox.SizeMode = "Zoom"
                $pictureBox.Width =  $img.Size.Width;
                $pictureBox.Height =  $img.Size.Height;
                $pictureBox.Image = $img;
                ### HYPERLINK WHEN CLICKING PICTURE ###
                <#
                If ($img -and $Hyperlink)
                {
                    $pictureBox.Add_Click({
                                ### Opens up IE
                                $ie = New-Object -ComObject InternetExplorer.Application
                                $ie.Navigate($Hyperlink)
                                $ie.Visible = $true
                    })
                }#>
                $form.controls.add($pictureBox)
                $form.Add_Shown( { $form.Activate() } )
                $form.ShowDialog()
            }
            If ($Hyperlink)
            {
                                $ie = New-Object -ComObject InternetExplorer.Application
                                $ie.Navigate($Hyperlink)
                                $ie.Visible = $true
            }
            $window.Close()
        })
    $TitleLabel.Content = $Title
    $MessageTB.Text = $Message
    $window.Left = $([System.Windows.SystemParameters]::WorkArea.Width-$window.Width)
    $window.Top = $([System.Windows.SystemParameters]::WorkArea.Height-$window.Height)
$timer = new-object System.Windows.Forms.Timer
$timer.Interval = 20000
$timer.Add_Tick({
 $timer.Stop();
    #$timer.Tick -= new EventHandler(formClose_Tick);
$window.Close()
})
$timer.Start()
$window.ShowDialog() | Out-Null
}

我已经写了以下时间(它在远程机器上工作 - 必须在吐司消息工作之前在机器上签名,但这没关系,但这没关系如果控制台被锁定 - 只需要签名)即可。(我不应该发布此内容,因为主题启动器没有提供他/她自己的代码,但这次我会例外)。请记住,我在此脚本中有一个可变$ CRED,该脚本未设置为参数(此变量包含管理凭据,因为启用了远程计算机)。

Function Invoke-ToastMessage
{
[CmdletBinding()]
    Param
        (
        [string]$Message,
        [string]$Notifier = "Administrators",
        [string]$ComputerName = $null
        )
[scriptblock]$ToastScriptRemote = {
$Message = $args[0]
$Notifier = $args[1]
# XML Template
[xml]$XmlTemplate = @"
<toast scenario="reminder">
  <visual>
    <binding template="ToastGeneric">
      <text>Admin Notification</text>
      <text>$Message</text>
    </binding>
  </visual>
  <actions>
  </actions>
</toast>
"@
    # fake load the assemblies
    [void][Windows.UI.Notifications.ToastNotification,Windows.UI.Notifications,ContentType=WindowsRuntime]
    [void][Windows.Data.Xml.Dom.XmlDocument,Windows.Data.Xml.Dom,ContentType=WindowsRuntime]
    $FinalXML = [Windows.Data.Xml.Dom.XmlDocument]::new()
    $FinalXML.LoadXml($XmlTemplate.OuterXml)
    ## create the toast
    $Toast = [Windows.UI.Notifications.ToastNotification]::new($FinalXML)
    ## Show the TOAST message
    [Windows.UI.Notifications.ToastNotificationManager]::CreateToastNotifier($Notifier).show($Toast)
    }
[scriptblock]$ToastScriptLocal = {
# XML Template
[xml]$XmlTemplate = @"
<toast scenario="reminder">
  <visual>
    <binding template="ToastGeneric">
      <text>Admin Notification</text>
      <text>$Message</text>
    </binding>
  </visual>
  <actions>
  </actions>
</toast>
"@
    # fake load the assemblies
    [void][Windows.UI.Notifications.ToastNotification,Windows.UI.Notifications,ContentType=WindowsRuntime]
    [void][Windows.Data.Xml.Dom.XmlDocument,Windows.Data.Xml.Dom,ContentType=WindowsRuntime]
    $FinalXML = [Windows.Data.Xml.Dom.XmlDocument]::new()
    $FinalXML.LoadXml($XmlTemplate.OuterXml)
    ## create the toast
    $Toast = [Windows.UI.Notifications.ToastNotification]::new($FinalXML)
    ## Show the TOAST message
    [Windows.UI.Notifications.ToastNotificationManager]::CreateToastNotifier($Notifier).show($Toast)
    }
if (![string]::IsNullOrEmpty($ComputerName))
    {
        Invoke-Command -ComputerName $ComputerName -Credential $cred -ScriptBlock $ToastScriptRemote -ArgumentList $Message,$Notifier
    }
    else {$ToastScriptLocal.Invoke()}
}