使用PowerShell版本2的WPF运行空间崩溃

时间:2016-03-07 09:26:59

标签: wpf powershell runspace

我正在写一个powershell脚本,它安装了各种驱动程序。

我的脚本工作正常,所以我想使用WPF添加一个gui。 首先,我使用WPF创建了一个gui,没什么了不起的,只是一个带标签的窗口。

我想从我的安装脚本更新此标签。所以我创建了两个运行空间,一个用于创建和显示WPF gui,另一个用于执行我的安装脚本。 只要我使用PowerShell版本3或更高版本,这样就可以正常工作。使用PowerShell 2,我必须使用新的Windows 7安装,wpf运行空间崩溃。

我希望有一种方法可以使用PowerShell版本2。

这是一个示例脚本,展示了我正在做的事情。

#########################################################################################
#
#   W P F - R U N S P A C E
#
#########################################################################################
$syncHashWpfLuaNotification = [hashtable]::Synchronized(@{})
$runspaceWpfLuaNotification =[runspacefactory]::CreateRunspace()
$runspaceWpfLuaNotification.ApartmentState = "STA"
$runspaceWpfLuaNotification.ThreadOptions = "ReuseThread"         
$runspaceWpfLuaNotification.Open()
$runspaceWpfLuaNotification.SessionStateProxy.SetVariable("syncHashWpfLuaNotification",$syncHashWpfLuaNotification)          
$psCmd = [PowerShell]::Create().AddScript({   
    [void][System.Reflection.Assembly]::LoadWithPartialName('presentationframework')
    [xml]$xaml = @"
<Window 
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        Title="TreiberInstaller" Height="431" Width="626" Background="Black"
        WindowStyle="None" ResizeMode="NoResize" WindowStartupLocation="CenterScreen">
    <Grid>
        <Grid.ColumnDefinitions>
            <ColumnDefinition Width="1*" />
            <ColumnDefinition Width="2*" />
            <ColumnDefinition Width="1*" />
        </Grid.ColumnDefinitions>
        <Grid.RowDefinitions>
            <RowDefinition Height="2*" />
            <RowDefinition Height="1*" />
            <RowDefinition Height="1*" />
        </Grid.RowDefinitions>

        <Label Name="lblProgress" Content="Progress"  HorizontalAlignment="Center" Grid.Column="1" Grid.Row="2" VerticalAlignment="Top" Foreground="White" FontFamily="Calibri" FontSize="18" HorizontalContentAlignment="Center"/>

    </Grid>
</Window>
"@
    $reader=(New-Object System.Xml.XmlNodeReader $xaml)
    $syncHashWpfLuaNotification.Window = [Windows.Markup.XamlReader]::Load( $reader )
    $syncHashWpfLuaNotification.lblProgress = $syncHashWpfLuaNotification.window.FindName("lblProgress")
    $syncHashWpfLuaNotification.Window.ShowDialog() | Out-Null
    $syncHashWpfLuaNotification.Error = $Error
})
$psCmd.Runspace = $runspaceWpfLuaNotification
$data = $psCmd.BeginInvoke()
Sleep -Milliseconds 450 # Wait a moment that the gui is ready




#########################################################################################
#
#   W O R K E R - R U N S P A C E
#
#########################################################################################
$ScriptBlock = { 
    #----------------------------------------------------------------------
    #   SetLabelText: Sets the lable-text
    #----------------------------------------------------------------------
    function SetLabelText
    {
        param(
                [Parameter(Position=0, Mandatory = $true, ValueFromPipeline = $false)]
                [ValidateNotNullOrEmpty()]
                [string]$Text
            )

        if(-not $syncHashWpfLuaNotification) {return}

        try
        {
            $syncHashWpfLuaNotification.Window.Dispatcher.invoke(
                    [action]{$syncHashWpfLuaNotification.lblProgress.Content = $Text},
                    "Normal"
            )
        }
        catch {}
    }

    #----------------------------------------------------------------------
    #   CloseProgressWindow: Closes the window
    #----------------------------------------------------------------------
    function CloseProgressWindow()
    {
        if(-not $syncHashWpfLuaNotification) {return}

        try
        {
            $syncHashWpfLuaNotification.Window.Dispatcher.invoke(
                    [action]{$syncHashWpfLuaNotification.Window.Close()},
                    "Normal"
            )
        }
        catch{}
    }



    #Starting here
    SetLabelText -Text "Starting installation..."

    Sleep 2

    for($i=1;$i -le 19; $i++)
    {
        SetLabelText -Text ("Progress Step " + $i)
    }

    for($i=20;$i -le 24; $i++)
    {
        SetLabelText -Text ("Progress Step " + $i)
        Sleep 1
    }

    CloseProgressWindow
} #End of $ScriptBlock


$syncHash1 = [hashtable]::Synchronized(@{})
$workerRunspace =[runspacefactory]::CreateRunspace()
$workerRunspace.ApartmentState = "STA"
$workerRunspace.ThreadOptions = "ReuseThread"         
$workerRunspace.Open()
$workerRunspace.SessionStateProxy.SetVariable("syncHash1",$syncHash1)          
$workerRunspace.SessionStateProxy.SetVariable("syncHashWpfLuaNotification",$syncHashWpfLuaNotification)          
$psCmd1 = [PowerShell]::Create().AddScript($ScriptBlock)
$psCmd1.Runspace = $workerRunspace
$data = $psCmd1.BeginInvoke()






#########################################################################################
#
#   S C R I P T   E N D   
#
#########################################################################################

#Wait for end of both runspaces
while(($runspaceWpfLuaNotification.RunspaceAvailability -eq "Busy") -or ($workerRunspace.RunspaceAvailability -eq "Busy"))
{
    if($runspaceWpfLuaNotification.RunspaceAvailability -eq "Busy") { Write-Host "Window is open" }
    if($workerRunspace.RunspaceAvailability -eq "Busy") { Write-Host "Worker is running" }

    Sleep 1
}

Write-Host "Script ended"

2 个答案:

答案 0 :(得分:1)

您的通行证委托给Dispatcher.Invoke的方式存在一些问题。

  • ScriptBlock文字与当前会话状态有关。如果您将有界ScriptBlock传递给不同的Runspace,则可能会导致一些不受欢迎的影响,例如目标Runspace中未调用的代码或死锁。请参阅my other answer。所以,首先要做的是创建一个没有界限的新ScriptBlock。您可以使用[ScriptBlock]::Create方法执行此操作。
  • 由于ScriptBlock不再受当前会话状态的限制,因此您无法从代码中引用变量,就像在代码中使用$Text一样。您应该将它们作为附加参数传递给Dispatcher.Invoke
  • 在将与PowerShell v2一起使用的.NET Framework 3.5中,Invoke(Action, DispatcherPriority)类中没有Dispatcher重载,因此Invoke([Action]{...}, "Normal")将被解析为Invoke(Delegate, Object[])而是方法。您应该使用不同的重载:Invoke(DispatcherPriority, Delegate),它存在于.NET Framework 3.5中。

答案 1 :(得分:0)

尝试下载适用于PS V2的WPFRunspace。它为WPF和基于表单的脚本提供后台工作程序。