2016-11-23 4 views
-1

Текущая ситуация: Я создал сценарий 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 

} 
+0

Хорошо, вы можете поделиться сценарием? как вы ссылаетесь на тост-уведомление? – 4c74356b41

ответ

0

Я написал следующее время Loong назад (он работает на удаленных компьютерах - пользователь должен быть подписан в на машине до тоста сообщения работают, но не имеет значения, заблокирована ли консоль - они просто должны быть подписаны). (Я не должен публиковать это, поскольку стартер темы не предоставил свой собственный код, но на этот раз я сделаю исключение). Имейте в виду, что у меня есть переменная $ cred в этом скрипте, которая не задана как параметр (эта переменная содержит учетные данные администратора из-за Invoke-Command на удаленную машину).

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()} 
} 
+0

Привет. Это работает? Потому что я знаю, что вызов invoke-команды запускает скрипты в порядке, но они не позволяют сценариям, отображающим графические интерфейсы, всплывать –

+0

@bluff Я просто протестировал его! Он работает. Вау. Я действительно думал, что вы не сможете запускать удаленные сценарии с помощью графических интерфейсов. есть ли способ изменить тост, который появляется так, что мы можем иметь кнопку, которая может открыть веб-страницу или показать изображение и т. д.? Благодаря! –

+0

Вы можете вставлять изображения (UNC-ресурсы не работают, подключенные диски действительно работают) и используют встроенные звуки Windows. Добавление кнопок также возможно, но поскольку вы не можете подключить их к чему-то, они довольно бесполезны. См. Документацию по https://msdn.microsoft.com/en-us/windows/uwp/controls-and-patterns/tiles-and-notifications-adaptive-interactive-toasts для получения информации о том, как сформировать XML для получения изображений и т.д. в них. – bluuf

Смежные вопросы