Launch-GetMailboxGUI.ps1


<#PSScriptInfo
 
.VERSION 1.0.2
 
.GUID b5592ee1-dc40-4e2e-aa14-a819b595ceb4
 
.AUTHOR sammy
 
.DESCRIPTION
 test
 
.NOTES
    This is a script to demonstrate the usage of Graphical User Interface (GUI) with PowerShell.
 
    This script requires:
        - PowerShell v3 minium
        - Exchange Management Tools (or being logged on Exchange Online on the current PowerShell session)
     
    These above dependencies are checked when the script launches.
 
.LINK
 https://github.com/SammyKrosoft/Exchange-Get-Mailboxes-GUI
 
.COMPANYNAME
 
.COPYRIGHT
 
.TAGS
 
.LICENSEURI
 
.PROJECTURI
 
.ICONURI
 
.EXTERNALMODULEDEPENDENCIES
 
.REQUIREDSCRIPTS
 
.EXTERNALSCRIPTDEPENDENCIES
 
.RELEASENOTES
 
 
.PRIVATEDATA
 
#>


<#
 
 
#>
 

$Version = "1.0.2"
<#Version History
v1.00 - added PSSCriptInfo for publishing on PSGallery
v0.97 - made window bigger - added Copy to Clipboard button for mailboxes list - added ServerName and database info
v0.96 - added icon, made window a bit bigger
v0.95 - fixed Arbitration mailbox SIR and Quota functions
v0.92 - Added Arbitration mailbox check box
v0.91 - Added ability to sort each columns in List quota action ...
#>

#region FUNCTIONS other than Form events
Function IsPSV3 {
    <#
    .DESCRIPTION
    Just printing Powershell version and returning "true" if powershell version
    is Powershell v3 or more recent, and "false" if it's version 2.
    .OUTPUTS
    Returns $true or $false
    .EXAMPLE
    IsPSVersionV3
    #>

    $PowerShellMajorVersion = $PSVersionTable.PSVersion.Major
    $msgPowershellMajorVersion = "You're running Powershell v$PowerShellMajorVersion"
    Write-Host $msgPowershellMajorVersion -BackgroundColor blue -ForegroundColor yellow
    If($PowerShellMajorVersion -le 2){
        Write-Host "Sorry, PowerShell v3 or more is required. Exiting."
        Return $false
        Exit
    } Else {
        Write-Host "You have PowerShell v3 or later, great !" -BackgroundColor blue -ForegroundColor yellow
        Return $true
        }
}

Function Test-ExchTools(){
    <#
    .SYNOPSIS
    This small function will just check if you have Exchange tools installed or available on the
    current PowerShell session.
 
    .DESCRIPTION
    The presence of Exchange tools are checked by trying to execute "Get-ExBanner", one of the basic Exchange
    cmdlets that runs when the Exchange Management Shell is called.
 
    Just use Test-ExchTools in your script to make the script exit if not launched from an Exchange
    tools PowerShell session...
 
    .EXAMPLE
    Test-ExchTools
    => will exit the script/program si Exchange tools are not installed
    #>

    Try
    {
        Get-command Get-MAilbox -ErrorAction Stop
        $ExchInstalledStatus = $true
        $Message = "Exchange tools are present !"
        Write-Host $Message -ForegroundColor Blue -BackgroundColor Red
    }
    Catch [System.SystemException]
    {
        $ExchInstalledStatus = $false
        $Message = "Exchange Tools are not present ! This script/tool need these. Exiting..."
        Write-Host $Message -ForegroundColor red -BackgroundColor Blue
        # Add-Type -AssemblyName presentationframework, presentationcore
        # Option #4 - a message, a title, buttons, and an icon
        # More info : https://msdn.microsoft.com/en-us/library/system.windows.messageboximage.aspx
        $msg = "You must run this tool from an Exchange-enabled PowerShell console like Exchange Management Console or a PowerShell session where you imported an Exchange session."
        $Title = "Error - No Exchange Tools available !"
        $Button = "Ok"
        $Icon = "Error"
        [System.Windows.MessageBox]::Show($msg,$Title, $Button, $icon)
        Exit
    }
    Return $ExchInstalledStatus
}

Function Run-Action{
    $SelectedAction = $wpf.comboSelectAction.SelectedItem.Content
    Switch ($SelectedAction) {
        #Region Disable Mailbox ***********************************************************
        "Disable Mailbox"  {
            Write-host "Displaying Info"
            Write-Host "Listing selected mailbox names:"
            $SelectedITems = $wpf.GridView.SelectedItems
            $List = @()
            $SelectedItems | Foreach{
                $List += ("""") + $($_.Alias) + ("""")
            }
            $List = $List -join ","
            $Command = "$List | Disable-Mailbox"
            WRite-Host "About to execute action on $($SelectedItems.Count) mailboxes..."
            Write-Host "About to run $Command"
        }
        #endregion
        #End of the Disabled Mailbox region
        #region List Quotas
        "List Quotas"   {
            Write-host "Displaying Mailbox SIR and retention settings status"
            $SelectedITems = $wpf.GridView.SelectedItems
            Write-host "Displaying Mailbox Single Item Recovery and retention settings status for $($SelectedItems.count) items..."
            $List = @()
            $SelectedItems | Foreach {
                $List += $_.primarySMTPAddress.tostring()
            }
            #$List = $List -join ","
            Function Get-MailboxQuotas {
                [CmdLetBinding()]
                Param(
                    [Parameter(Mandatory = $False, Position = 1)][string[]]$List,
                    [Parameter(Mandatory = $False, Position = 2)][switch]$Arbitration
                )
                #Initiating stopwatch to measure the time it takes to retrieve mailboxes
                $stopwatch = [system.diagnostics.stopwatch]::StartNew()
                If ($Arbitration){
                    $QueryMailboxFeaturesStd = $List | get-mailbox -Arbitration | Select DisplayName,Name,ServerName,Database,*quota*,OrganizationalUnit
                } Else {
                    $QueryMailboxFeaturesStd = $List | get-mailbox | Select DisplayName,Name,ServerName,Database,*quota*,OrganizationalUnit
                }
                
                $QueryMailboxFeatures = @()
                Foreach ($mailbox in $QueryMailboxFeaturesStd){
                    $objTemp = $mailbox
                    if ($($objTemp.OrganizationalUnit) -match "prod.outlook.com"){
                        $CloudMailbox = $True
                    }else{
                        $CloudMailbox = $false
                    }
                    #<optional> - Removing OrganizationalUnit information ...
                    $ObjTemp = $ObjTemp | Select DisplayName, Database, UseDatabaseQuotaDefaults, ProhibitSendQuota, ProhibitSendReceiveQuota, IssueWarningQuota, RulesQuota, CalendarLoggingQuota, RecoverableItemsQuota, RecoverableItemsWarningQuota, ArchiveQuota, ArchiveWarningQuota
                    if ($CloudMailbox){
                        $objTemp | add-Member -NotePropertyName DatabaseProhibitSRQuota -NotePropertyValue "Cloud mailbox (no DB Quota info)"
                        $ObjTemp | Add-Member -NotePropertyName DatabaseSendQuota -NotePropertyValue "Cloud mailbox (no DB Quota info)"
                        $ObjTemp | Add-Member -NotePropertyName DatabaseWarningQuota -NotePropertyValue "Cloud mailbox (no DB Quota info)"
                    } Else {
                        $objTemp | add-Member -NotePropertyName DatabaseProhibitSRQuota -NotePropertyValue $((Get-MailboxDatabase $($Mailbox.Database)).ProhibitSendReceiveQuota)
                        $ObjTemp | Add-Member -NotePropertyName DatabaseSendQuota -NotePropertyValue $((Get-MailboxDatabase $($Mailbox.Database)).ProhibitSendQuota)
                        $ObjTemp | Add-Member -NotePropertyName DatabaseWarningQuota -NotePropertyValue $((Get-MailboxDatabase $($Mailbox.Database)).IssueWarningQuota)
                    }
                    $QueryMailboxFeatures += $objTemp
                }
                [System.Collections.IENumerable]$MailboxFeatures = @($QueryMailboxFeatures)
                Write-host $($MailboxFeatures | ft | out-string)
                
                #Stopping stopwatch
                $stopwatch.Stop()
                $msg = "`n`nInstruction took $([math]::round($($StopWatch.Elapsed.TotalSeconds),2)) seconds ..."
                Write-Host $msg
                $msg = $null
                $StopWatch = $null

                #region Get-MailboxFeaturesView Form definition
                # Load a WPF GUI from a XAML file build with Visual Studio
                Add-Type -AssemblyName presentationframework, presentationcore
                $wpf = @{ }
                # NOTE: Either load from a XAML file or paste the XAML file content in a "Here String"
                #$inputXML = Get-Content -Path ".\WPFGUIinTenLines\MainWindow.xaml"
                $inputXML = @"
                <Window x:Name="frmMbxQuotaStatus" x:Class="Get_CASMAilboxFeaturesWPF.MainWindow"
                                        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
                                        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
                                        xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
                                        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
                                        xmlns:local="clr-namespace:Get_CASMAilboxFeaturesWPF"
                                        mc:Ignorable="d"
                                        Title="Mailboxes Quotas settings status" Height="450" Width="872.145" ResizeMode="NoResize">
                    <Grid>
                        <DataGrid x:Name="DataGrid" HorizontalAlignment="Left" Height="326" Margin="10,10,-59,0" VerticalAlignment="Top" Width="844" IsReadOnly="True"/>
                        <Button x:Name="btnClose" Content="Close" HorizontalAlignment="Left" Margin="748,352,0,0" VerticalAlignment="Top" Width="106" Height="46"/>
                        <Button x:Name="btnClipboard" Content="Copy to clipboard" HorizontalAlignment="Left" Margin="10,352,0,0" VerticalAlignment="Top" Width="174" Height="46"/>
 
                    </Grid>
                </Window>
"@


                $inputXMLClean = $inputXML -replace 'mc:Ignorable="d"','' -replace "x:N",'N' -replace 'x:Class=".*?"','' -replace 'd:DesignHeight="\d*?"','' -replace 'd:DesignWidth="\d*?"',''
                [xml]$xaml = $inputXMLClean
                $reader = New-Object System.Xml.XmlNodeReader $xaml
                $tempform = [Windows.Markup.XamlReader]::Load($reader)
                $namedNodes = $xaml.SelectNodes("//*[@*[contains(translate(name(.),'n','N'),'Name')]]")
                $namedNodes | ForEach-Object {$wpf.Add($_.Name, $tempform.FindName($_.Name))}

                #Get the form name to be used as parameter in functions external to form...
                $FormName = $NamedNodes[0].Name


                #Define events functions
                #region Load, Draw (render) and closing form events
                #Things to load when the WPF form is loaded aka in memory
                $wpf.$FormName.Add_Loaded({
                    #Update-Cmd
                    $wpf.DataGrid.ItemsSource = $MailboxFeatures
                    $wpf.DataGrid.Columns | Foreach {
                        $_.CanUserSort = $true
                    }            
                })
                #Things to load when the WPF form is rendered aka drawn on screen
                $wpf.$FormName.Add_ContentRendered({
                    #Update-Cmd
                })
                $wpf.$FormName.add_Closing({
                    $msg = "Closed the MBX SIR and retention settings status list window"
                    write-host $msg
                })
                $wpf.btnClipboard.add_Click({
                    $CSVClip = $mailboxFeatures | ConvertTo-CSV -NoTypeInformation
                    $CSVClip | clip.exe
                    $title = "Copied !"
                    $msg = "Data copied to the clipboard ! `n`rUse CTRL+V on Notepad or on Excel !"
                    [System.Windows.MessageBox]::Show($msg,$title, "OK","Asterisk")
                })
                $wpf.btnClose.add_Click({
                    $wpf.$FormName.Close()
                })

                #endregion Load, Draw and closing form events
                #End of load, draw and closing form events

                #HINT: to update progress bar and/or label during WPF Form treatment, add the following:
                # ... to re-draw the form and then show updated controls in realtime ...
                $wpf.$FormName.Dispatcher.Invoke("Render",[action][scriptblock]{})


                # Load the form:
                # Older way >>>>> $wpf.MyFormName.ShowDialog() | Out-Null >>>>> generates crash if run multiple times
                # Newer way >>>>> avoiding crashes after a couple of launches in PowerShell...
                # USing method from https://gist.github.com/altrive/6227237 to avoid crashing Powershell after we re-run the script after some inactivity time or if we run it several times consecutively...
                $async = $wpf.$FormName.Dispatcher.InvokeAsync({
                    $wpf.$FormName.ShowDialog() | Out-Null
                })
                $async.Wait() | Out-Null

                #endregion
                # end of Form definition for Get-MailboxFeaturesView
                
            }

            if ($wpf.chkArbitrationOnly.IsChecked){
                Get-MailboxQuotas $List -Arbitration
            } Else {
                Get-MailboxQuotas $List
            }
        }
        #endregion
        #End of the List Quotas region
        #region Single Item Recovery Status
        "List Single Item Recovery status" {
            Write-host "Displaying Mailbox SIR and retention settings status"
            $SelectedITems = $wpf.GridView.SelectedItems
            Write-host "Displaying Mailbox Single Item Recovery and retention settings status for $($SelectedItems.count) items..."
            $List = @()
            $SelectedItems | Foreach {
                $List += $_.primarySMTPAddress.tostring()
            }
            #$List = $List -join ","
            Function Get-MailboxSIRView {
                [CmdLetBinding()]
                Param(
                    [Parameter(Mandatory = $False, Position = 1)][string[]]$List,
                    [Parameter(Mandatory = $False, Position = 2)][switch]$Arbitration
                )
                #Initiating stopwatch to measure the time it takes to retrieve mailboxes
                $stopwatch = [system.diagnostics.stopwatch]::StartNew()
                If ($Arbitration){
                    $QueryMailboxFeatures = $List | get-mailbox -Arbitration | Select DisplayName,Name,*item*,OrganizationalUnit
                } Else {
                    $QueryMailboxFeatures = $List | Get-Mailbox | Select DisplayName,Name, *item*, OrganizationalUnit
                }
                [System.Collections.IENumerable]$MailboxFeatures = @($QueryMailboxFeatures)
                Write-host $($MailboxFeatures | ft | out-string)
                
                #Stopping stopwatch
                $stopwatch.Stop()
                $msg = "`n`nInstruction took $([math]::round($($StopWatch.Elapsed.TotalSeconds),2)) seconds ..."
                Write-Host $msg
                $msg = $null
                $StopWatch = $null

                #region Get-MailboxFeaturesView Form definition
                # Load a WPF GUI from a XAML file build with Visual Studio
                Add-Type -AssemblyName presentationframework, presentationcore
                $wpf = @{ }
                # NOTE: Either load from a XAML file or paste the XAML file content in a "Here String"
                #$inputXML = Get-Content -Path ".\WPFGUIinTenLines\MainWindow.xaml"
                $inputXML = @"
                <Window x:Name="frmMbxSIRStatus" x:Class="Get_CASMAilboxFeaturesWPF.MainWindow"
                                        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
                                        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
                                        xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
                                        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
                                        xmlns:local="clr-namespace:Get_CASMAilboxFeaturesWPF"
                                        mc:Ignorable="d"
                                        Title="Mailboxes Single Item Recovery and Retention settings status" Height="450" Width="872.145" ResizeMode="NoResize">
                    <Grid>
                        <DataGrid x:Name="DataGridCASMbx" HorizontalAlignment="Left" Height="326" Margin="10,10,-59,0" VerticalAlignment="Top" Width="844" IsReadOnly="True"/>
                        <Button x:Name="btnClose" Content="Close" HorizontalAlignment="Left" Margin="748,352,0,0" VerticalAlignment="Top" Width="106" Height="46"/>
                        <Button x:Name="btnClipboard" Content="Copy to clipboard" HorizontalAlignment="Left" Margin="10,352,0,0" VerticalAlignment="Top" Width="174" Height="46"/>
 
                    </Grid>
                </Window>
"@


                $inputXMLClean = $inputXML -replace 'mc:Ignorable="d"','' -replace "x:N",'N' -replace 'x:Class=".*?"','' -replace 'd:DesignHeight="\d*?"','' -replace 'd:DesignWidth="\d*?"',''
                [xml]$xaml = $inputXMLClean
                $reader = New-Object System.Xml.XmlNodeReader $xaml
                $tempform = [Windows.Markup.XamlReader]::Load($reader)
                $namedNodes = $xaml.SelectNodes("//*[@*[contains(translate(name(.),'n','N'),'Name')]]")
                $namedNodes | ForEach-Object {$wpf.Add($_.Name, $tempform.FindName($_.Name))}

                #Get the form name to be used as parameter in functions external to form...
                $FormName = $NamedNodes[0].Name


                #Define events functions
                #region Load, Draw (render) and closing form events
                #Things to load when the WPF form is loaded aka in memory
                $wpf.$FormName.Add_Loaded({
                    #Update-Cmd
                    $wpf.DataGridCASMbx.ItemsSource = $MailboxFeatures
                })
                #Things to load when the WPF form is rendered aka drawn on screen
                $wpf.$FormName.Add_ContentRendered({
                    #Update-Cmd
                })
                $wpf.$FormName.add_Closing({
                    $msg = "Closed the MBX SIR and retention settings status list window"
                    write-host $msg
                })
                $wpf.btnClipboard.add_Click({
                    $CSVClip = $mailboxFeatures | ConvertTo-CSV -NoTypeInformation
                    $CSVClip | clip.exe
                    $title = "Copied !"
                    $msg = "Data copied to the clipboard ! `n`rUse CTRL+V on Notepad or on Excel !"
                    [System.Windows.MessageBox]::Show($msg,$title, "OK","Asterisk")
                })
                $wpf.btnClose.add_Click({
                    $wpf.$FormName.Close()
                })

                #endregion Load, Draw and closing form events
                #End of load, draw and closing form events

                #HINT: to update progress bar and/or label during WPF Form treatment, add the following:
                # ... to re-draw the form and then show updated controls in realtime ...
                $wpf.$FormName.Dispatcher.Invoke("Render",[action][scriptblock]{})


                # Load the form:
                # Older way >>>>> $wpf.MyFormName.ShowDialog() | Out-Null >>>>> generates crash if run multiple times
                # Newer way >>>>> avoiding crashes after a couple of launches in PowerShell...
                # USing method from https://gist.github.com/altrive/6227237 to avoid crashing Powershell after we re-run the script after some inactivity time or if we run it several times consecutively...
                $async = $wpf.$FormName.Dispatcher.InvokeAsync({
                    $wpf.$FormName.ShowDialog() | Out-Null
                })
                $async.Wait() | Out-Null

                #endregion
                # end of Form definition for Get-MailboxFeaturesView
                
            }

            If ($wpf.chkArbitrationOnly.IsChecked){
                Get-MailboxSIRView $List -Arbitration
            } Else {
                Get-MailboxSIRView $List
            }
            
        }
        #endregion
        #End of the Single Item Recovery status region
        #region List Mailbox Features
        "List Mailbox Features"  {
            if ($wpf.chkArbitrationOnly.IsChecked){
                # Option #4 - a message, a title, buttons, and an icon
                # More info : https://msdn.microsoft.com/en-us/library/system.windows.messageboximage.aspx
                $msg = "Arbitration mailboxes is checked - cannot get mailbox features for Arbitration mailboxes"
                $Title = "Error - Arbitration mailboxes don't get features"
                $Button = "Ok"
                $Icon = "Error"
                [System.Windows.MessageBox]::Show($msg,$Title, $Button, $icon)
                Return
            }
            Write-host "Displaying Mailbox Features"
            $SelectedITems = $wpf.GridView.SelectedItems
            Write-host "Displaying Mailbox Features for $($SelectedItems.count) items..."
            $List = @()
            $SelectedItems | Foreach {
                $List += $_.primarySMTPAddress.tostring()
            }
            #$List = $List -join ","
            Function Get-MailboxFeaturesView {
                [CmdLetBinding()]
                Param(
                    [Parameter(Mandatory = $False, Position = 1)][string[]]$List
                )

                #Initiating stopwatch to measure the time it takes to retrieve mailboxes
                $stopwatch = [system.diagnostics.stopwatch]::StartNew()

                $QueryMailboxFeatures = $List | Get-CASMAilbox | Select DisplayName, *enabled, *MAPIblock*
                [System.Collections.IENumerable]$MailboxFeatures = @($QueryMailboxFeatures)
                Write-host $($MailboxFeatures | ft DisplayName, ActiveSyncEnabled,OWAEnabled,ECPEnabled,MAPIEnabled,MAPIBlockOutlookRpcHttp,MapiHttpEnabled  -a | out-string)

                #Stopping stopwatch
                $stopwatch.Stop()
                $msg = "`n`nInstruction took $([math]::round($($StopWatch.Elapsed.TotalSeconds),2)) seconds..."
                Write-Host $msg
                $msg = $null
                $StopWatch = $null

                #region Get-MailboxFeaturesView Form definition
                # Load a WPF GUI from a XAML file build with Visual Studio
                Add-Type -AssemblyName presentationframework, presentationcore
                $wpf = @{ }
                # NOTE: Either load from a XAML file or paste the XAML file content in a "Here String"
                #$inputXML = Get-Content -Path ".\WPFGUIinTenLines\MainWindow.xaml"
                $inputXML = @"
                <Window x:Name="frmCASMBOXProps" x:Class="Get_CASMAilboxFeaturesWPF.MainWindow"
                                        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
                                        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
                                        xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
                                        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
                                        xmlns:local="clr-namespace:Get_CASMAilboxFeaturesWPF"
                                        mc:Ignorable="d"
                                        Title="Mailbox features enabled and blocked status" Height="450" Width="872.145" ResizeMode="NoResize">
                    <Grid>
                        <DataGrid x:Name="DataGridCASMbx" HorizontalAlignment="Left" Height="326" Margin="10,10,-59,0" VerticalAlignment="Top" Width="844" IsReadOnly="True"/>
                        <Button x:Name="btnClose" Content="Close" HorizontalAlignment="Left" Margin="748,352,0,0" VerticalAlignment="Top" Width="106" Height="46"/>
                        <Button x:Name="btnClipboard" Content="Copy to clipboard" HorizontalAlignment="Left" Margin="10,352,0,0" VerticalAlignment="Top" Width="174" Height="46"/>
 
                    </Grid>
                </Window>
"@


                $inputXMLClean = $inputXML -replace 'mc:Ignorable="d"','' -replace "x:N",'N' -replace 'x:Class=".*?"','' -replace 'd:DesignHeight="\d*?"','' -replace 'd:DesignWidth="\d*?"',''
                [xml]$xaml = $inputXMLClean
                $reader = New-Object System.Xml.XmlNodeReader $xaml
                $tempform = [Windows.Markup.XamlReader]::Load($reader)
                $namedNodes = $xaml.SelectNodes("//*[@*[contains(translate(name(.),'n','N'),'Name')]]")
                $namedNodes | ForEach-Object {$wpf.Add($_.Name, $tempform.FindName($_.Name))}

                #Get the form name to be used as parameter in functions external to form...
                $FormName = $NamedNodes[0].Name

                #Define events functions
                #region Load, Draw (render) and closing form events
                #Things to load when the WPF form is loaded aka in memory
                $wpf.$FormName.Add_Loaded({
                    #Update-Cmd
                    $wpf.DataGridCASMbx.ItemsSource = $MailboxFeatures
                })
                #Things to load when the WPF form is rendered aka drawn on screen
                $wpf.$FormName.Add_ContentRendered({
                    #Update-Cmd
                })
                $wpf.$FormName.add_Closing({
                    $msg = "Closed the MBX features list window"
                    write-host $msg
                })
                $wpf.btnClipboard.add_Click({
                    $CSVClip = $mailboxFeatures | ConvertTo-CSV -NoTypeInformation
                    $CSVClip | clip.exe
                    $title = "Copied !"
                    $msg = "Data copied to the clipboard ! `n`rUse CTRL+V on Notepad or on Excel !"
                    [System.Windows.MessageBox]::Show($msg,$title, "OK","Asterisk")
                })
                $wpf.btnClose.add_Click({
                    $wpf.$FormName.Close()
                })

                #endregion Load, Draw and closing form events
                #End of load, draw and closing form events

                #HINT: to update progress bar and/or label during WPF Form treatment, add the following:
                # ... to re-draw the form and then show updated controls in realtime ...
                $wpf.$FormName.Dispatcher.Invoke("Render",[action][scriptblock]{})


                # Load the form:
                # Older way >>>>> $wpf.MyFormName.ShowDialog() | Out-Null >>>>> generates crash if run multiple times
                # Newer way >>>>> avoiding crashes after a couple of launches in PowerShell...
                # USing method from https://gist.github.com/altrive/6227237 to avoid crashing Powershell after we re-run the script after some inactivity time or if we run it several times consecutively...
                $async = $wpf.$FormName.Dispatcher.InvokeAsync({
                    $wpf.$FormName.ShowDialog() | Out-Null
                })
                $async.Wait() | Out-Null

                #endregion
                # end of Form definition for Get-MailboxFeaturesView
                
            }

            Get-MailboxFeaturesView $List
        }
        #endregion
        #End of the List Mailbox Features region
    }
}

Function Update-Label ($msg) {
    $wpf.lblStatus.Content = $msg
    $Wpf.$FormName.Dispatcher.Invoke("Render",[action][scriptblock]{})
}

Function Working-Label {
        # Trick to enable a Label to update during work :
    # Follow with "Dispatcher.Invoke("Render",[action][scriptblobk]{})" or [action][scriptblock]::create({})
    $wpf.$FormName.IsEnabled = $False
    $wpf.lblStatus.Content = "Working ..."
    $wpf.lblStatus.ForeGround = [System.Windows.Media.Brushes]::Red
    $wpf.lblStatus.BackGround = [System.Windows.Media.Brushes]::Blue
    $Wpf.$FormName.Dispatcher.Invoke("Render",[action][scriptblock]{})
}

Function Ready-Label{
    $wpf.$FormName.IsEnabled = $True
    $wpf.lblStatus.Content = "Ready !"
    $wpf.lblStatus.ForeGround = [System.Windows.Media.Brushes]::Green
    $wpf.lblStatus.BackGround = [System.Windows.Media.Brushes]::Yellow
    $Wpf.$FormName.Dispatcher.Invoke("Render",[action][scriptblock]{})
}

Function Update-MainCommandLine {
    If ($wpf.chkArbitrationOnly.IsChecked){
        $commandLine = "Get-Mailbox -ResultSize Unlimited -Arbitration -ErrorAction Stop"
    } Else {
        If ($wpf.txtMailboxString.text -eq ""){
            $SearchSubstring = ("*")
        } Else {
            $SearchSubstring = ("*") + ($wpf.txtMailboxString.text) + ("*")
        }
        If ($wpf.chkUnlimited.IsChecked){
            $ResultSize = "Unlimited"
        } Else {
            $ResultSize = $wpf.txtResultSize.Text
        }
        $chkIncludeDiscovery = $false
        If ($chkIncludeDiscovery){
            $commandLine = "Get-Mailbox -ResultSize $ResultSize -Identity $SearchSubstring | Select Name,Alias,DisplayName,primarySMTPAddress,ServerName,Database"
        } Else {
            $commandLine = "Get-Mailbox -ResultSize $ResultSize -Identity $SearchSubstring -Filter {RecipientTypeDetails -ne `"DiscoveryMailbox`"} | Select Name,Alias,DisplayName,primarySMTPAddress,ServerName,Database"
        }
    }
    $wpf.txtMainCommand.Text = $CommandLine
}

$lblabout_Click = {
    $Language = "EN"
    switch ($Language)
    {
        "EN"
        {
            $systemst = "QXV0aG9yOiBTYW0gRHJleQ0Kc2FtZHJleUBtaWNyb3NvZnQuY29tDQpzYW1teUBob3RtYWlsLmZyDQpNaWNyb3NvZnQgRW`
        5naW5lZXIgc2luY2UgT2N0IDE5OTkNCjE5OTktMjAwMDogUHJlc2FsZXMgRW5naW5lZXIgKEZyYW5jZSkNCjIwMDAtMjAwMzogU3VwcG9yd`
        CBFbmdpbmVlciAoRnJhbmNlKQ0KMjAwMy0yMDA2OiB2ZXJ5IGZpcnN0IFBGRSBpbiBGcmFuY2UNCjIwMDYtMjAwOTogTUNTIENvbnN1bHRhb`
        nQgKEZyYW5jZSkNCjIwMDktMjAxMDogVEFNIChGcmFuY2UpDQoyMDEwLW5vdyA6IENvbnN1bHRhbnQgKENhbmFkYSkNCk11c2ljaWFuLCBjb`
        21wb3NlciAoS2V5Ym9hcmQsIEd1aXRhcikNClBsYW5lIHBpbG90IHNpbmNlIDE5OTUNCkZvciBTaGFyZWQgU2VydmljZXMgQ2FuYWRh"

        } 
        "FR"
        {
            $systemst = "QXV0ZXVyOiBTYW0gRHJleQ0Kc2FtZHJleUBtaWNyb3NvZnQuY29tDQpzYW1teUBob3RtYWlsLmZyDQpJbmfDqW5pZXVyIGNo`
        ZXogTWljcm9zb2Z0IGRlcHVpcyBPY3QgMTk5OQ0KMTk5OS0yMDAwOiBJbmfDqW5pZXVyIEF2YW50LVZlbnRlIChGcmFuY2UpDQoyMDAwLTIwMD`
        M6IFNww6ljaWFsaXN0ZSBUZWNobmlxdWUgKEZyYW5jZSkNCjIwMDMtMjAwNjogUHJlbWllciBQRkUgZW4gRnJhbmNlDQoyMDA2LTIwMDk6IENv`
        bnN1bHRhbnQgTUNTIChGcmFuY2UpDQoyMDA5LTIwMTA6IFJlc3BvbnNhYmxlIFRlY2huaXF1ZSBkZSBDb21wdGUgKEZyYW5jZSkNCjIwMTAtMjA`
        xNiA6IENvbnN1bHRhbnQgKENhbmFkYSkNCk11c2ljaWVuLCBjb21wb3NpdGV1ciAoQ2xhdmllciwgR3VpdGFyZSkNCkJyZXZldCBkZSBQaWxvdGU`
        gUHJpdsOpIGRlcHVpcyAxOTk1DQpQb3VyIFNlcnZpY2VzIFBhcnRhZ8OpcyBDYW5hZGE="

        }
    }
    $systemst = [System.Text.Encoding]::UTF8.GetString([System.Convert]::FromBase64String($systemst))
    # Option #4 - a message, a title, buttons, and an icon
    # More info : https://msdn.microsoft.com/en-us/library/system.windows.messageboximage.aspx
    $msg = $systemst
    $Title = $wpf.$FormName.Title
    $Button = "Ok"
    $Icon = "Information"
    [System.Windows.MessageBox]::Show($msg,$Title, $Button, $icon)
}


Function Get-Mailboxes {
    If ($wpf.chkUnlimited.IsChecked){
        Write-host "Unlimited specified ... ignoring Resultsize number..."
    } Else {
        If ($([int]$wpf.txtResultSize.Text) -gt 1000) {Write-Host "$($wpf.txtResultSize.Text) is greater than 1000 ..."} Else {write-host "$($wpf.txtResultSize.Text) is less than 1000"}
    }

    If ($([int]$wpf.txtResultSize.Text) -gt 1000 -or $wpf.chkUnlimited.IsChecked){
        # Option #4 - a message, a title, buttons, and an icon
        # More info : https://msdn.microsoft.com/en-us/library/system.windows.messageboximage.aspx
        if ($wpf.chkUnlimited.IsChecked) {
            $Specified = $wpf.chkUnlimited.Content
        } Else {
            $Specified = "$($wpf.txtResultSize.Text), which is more than 1000"
        }
        $msg = "WARNING: You specified -> $Specified <- for the Resultsize, mailbox collection can take a LOT of time, Continue ? (Y/N)"
        $Title = "Question..."
        $Button = "YesNo"
        $Icon = "Question"
        $Answer = [System.Windows.MessageBox]::Show($msg,$Title, $Button, $icon)
        If($Answer -eq "No"){Return}
    }
    Try {
        #Initiating stopwatch to measure the time it takes to retrieve mailboxes
        $stopwatch = [system.diagnostics.stopwatch]::StartNew()
        #Getting the command line from the text box where it's generated
        $commandLine = $wpf.txtMainCommand.text
        #Invoking the command line and storing in a variable
        $Mailboxes = invoke-expression $CommandLine
        $NewMailboxesObj = @()
        Foreach ($objTemp in $Mailboxes){
            If ($($objTemp.OrganizationalUnit) -match "prod.outlook.com"){
                $objtemp | Add-Member -NotePropertyName Location -NotePropertyValue "Cloud"
            } Else {
                $objtemp | Add-Member -NotePropertyName Location -NotePropertyValue "On-prem"
            }
            $NewMailboxesObj += $objtemp
        }
        $Mailboxes =  $NewMailboxesObj | Select Name,Alias,DisplayName,primarySMTPAddress, RecipientTypeDetails,Location,ServerName,Database
        #Stopping stopwatch
        $stopwatch.Stop()
        $msg = "`n`nInstruction took $([math]::round($($StopWatch.Elapsed.TotalSeconds),2)) seconds to retrieve $($Mailboxes.count) mailboxes..."
        Write-Host $msg
        $msg = $null
        $StopWatch = $null

        #Populating the GridView
        [System.Collections.IENumerable]$Results = @($Mailboxes)
        $wpf.GridView.ItemsSource = $Results
        $wpf.GridView.Columns | Foreach {
            $_.CanUserSort = $true
        }
        $wpf.lblNbItemsInGrid.Content = $($Results.Count)
    } Catch {
        $Mailboxes = $null
        $wpf.GridView.ItemsSource = $null
        write-host "ZERO MAILBOXES"
        $wpf.lblNbItemsInGrid.Content = 0
    }
}

#endregion

#========================================================
#region WPF form definition and load controls
#========================================================

# Load a WPF GUI from a XAML file build with Visual Studio
Add-Type -AssemblyName presentationframework, presentationcore
$wpf = @{}
# NOTE: Either load from a XAML file or paste the XAML file content in a "Here String"
# $inputXML = Get-Content -Path "C:\Users\Kamehameha\Documents\GitHub\PowerShell\Get-EventsFromEventLog\VisualStudio2017WPFDesign\Launch-EventsCollector-WPF\Launch-EventsCollector-WPF\MainWindow.xaml"
$inputXML = @"
<Window x:Name="WForm" x:Class="GridView_WPF.MainWindow"
        xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
        xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
        xmlns:d="http://schemas.microsoft.com/expression/blend/2008"
        xmlns:mc="http://schemas.openxmlformats.org/markup-compatibility/2006"
        xmlns:local="clr-namespace:GridView_WPF"
        mc:Ignorable="d"
        Title="Search Mailboxes" Height="722.39" Width="1026.5" ResizeMode="NoResize">
    <Grid>
        <DataGrid x:Name="GridView" HorizontalAlignment="Left" Height="425" Margin="353,10,0,0" VerticalAlignment="Top" Width="641"/>
        <TextBox x:Name="txtMailboxString" HorizontalAlignment="Left" Height="23" Margin="10,67,0,0" TextWrapping="Wrap" VerticalAlignment="Top" Width="338"/>
        <Label Content="Search for mailbox (substring of alias, e-mail address, &#xD;&#xA;display name, ...)" HorizontalAlignment="Left" VerticalAlignment="Top" Margin="10,11,0,0" Height="51" Width="302"/>
        <Button x:Name="btnRun" Content="Search" HorizontalAlignment="Left" Margin="10,95,0,0" VerticalAlignment="Top" Width="75" Height="32">
            <Button.Effect>
                <DropShadowEffect/>
            </Button.Effect>
        </Button>
        <Label x:Name="lblStatus" Content="Ready !" HorizontalAlignment="Left" VerticalAlignment="Top" Margin="10,658,0,0" Width="1010" FontStyle="Italic" FontWeight="Bold">
        </Label>
        <Button x:Name="btnAction" Content="Run on selected" Margin="360,596,0,0" IsEnabled="False" HorizontalAlignment="Left" VerticalAlignment="Top">
            <Button.Effect>
                <DropShadowEffect/>
            </Button.Effect>
        </Button>
        <ComboBox x:Name="comboSelectAction" HorizontalAlignment="Left" Margin="360,534,0,0" VerticalAlignment="Top" Height="30" SelectedIndex="0" IsEnabled="False" TextOptions.TextFormattingMode="Display">
            <ComboBox.Effect>
                <DropShadowEffect/>
            </ComboBox.Effect>
            <ComboBoxItem Content="List Mailbox Features"/>
            <ComboBoxItem Content="List Single Item Recovery status"/>
            <ComboBoxItem Content="List Quotas"/>
            <ComboBoxItem Content="Disable Mailbox"/>
        </ComboBox>
        <Label x:Name="lblNbItemsInGrid" Content="0" HorizontalAlignment="Left" VerticalAlignment="Top" Margin="506,440,0,0" Width="66"/>
        <Label Content="Number of Items in Grid:" HorizontalAlignment="Left" Margin="353,440,0,0" VerticalAlignment="Top" Width="148"/>
        <Label Content="Selected:" HorizontalAlignment="Left" Margin="353,471,0,0" VerticalAlignment="Top"/>
        <Label x:Name="lblNumberItemsSelected" Content="0" HorizontalAlignment="Left" Margin="460,471,0,0" VerticalAlignment="Top" Width="67"/>
        <TextBox x:Name="txtResultSize" HorizontalAlignment="Left" Height="23" Margin="224,98,0,0" TextWrapping="Wrap" VerticalAlignment="Top" Width="124" Text="100"/>
        <TextBlock HorizontalAlignment="Left" Margin="95,95,0,0" TextWrapping="Wrap" Text="ResultSize (aka Nb of mailboxes to display):" VerticalAlignment="Top" Width="124"/>
        <Label Content="Status:" HorizontalAlignment="Left" Margin="10,627,0,0" VerticalAlignment="Top"/>
        <TextBox x:Name="txtMainCommand" HorizontalAlignment="Left" Height="132" Margin="10,200,0,0" TextWrapping="Wrap" Text="Get-Mailbox command to be run..." VerticalAlignment="Top" Width="338" IsReadOnly="True"/>
        <Rectangle HorizontalAlignment="Left" Height="26" Margin="353,440,0,0" VerticalAlignment="Top" Width="232">
            <Rectangle.Stroke>
                <SolidColorBrush Color="{DynamicResource {x:Static SystemColors.ActiveBorderColorKey}}"/>
            </Rectangle.Stroke>
        </Rectangle>
        <Rectangle HorizontalAlignment="Left" Height="26" Margin="353,471,0,0" VerticalAlignment="Top" Width="174">
            <Rectangle.Stroke>
                <SolidColorBrush Color="{DynamicResource {x:Static SystemColors.ActiveBorderColorKey}}"/>
            </Rectangle.Stroke>
        </Rectangle>
        <Label Content="The command run when clicking on the Search button is:" HorizontalAlignment="Left" Margin="10,174,0,0" VerticalAlignment="Top" Width="338" FontStyle="Italic"/>
        <CheckBox x:Name="chkUnlimited" Content="Unlimited" HorizontalAlignment="Left" Margin="223,126,0,0" VerticalAlignment="Top"/>
        <Label x:Name="lblAbout" Content="." HorizontalAlignment="Left" Margin="999,0,0,0" VerticalAlignment="Top" Height="22" Width="21"/>
        <CheckBox x:Name="chkArbitrationOnly" Content="Arbitration only" HorizontalAlignment="Left" Margin="10,149,0,0" VerticalAlignment="Top"/>
        <Button x:Name="btnClipboard" Content="Copy all above items to clipboard" HorizontalAlignment="Left" Margin="590,442,0,0" VerticalAlignment="Top" Width="202" IsEnabled="False" Height="22">
            <Button.Effect>
                <DropShadowEffect/>
            </Button.Effect>
        </Button>
        <Label Content="#1 - Choose what you wish to see about the selected" HorizontalAlignment="Left" Margin="353,508,0,0" VerticalAlignment="Top"/>
        <Label Content="#2 - Run the above selected action on selected items" HorizontalAlignment="Left" Margin="353,570,0,0" VerticalAlignment="Top"/>
        <Border BorderBrush="{DynamicResource {x:Static SystemColors.ActiveBorderBrushKey}}" BorderThickness="1" HorizontalAlignment="Left" Height="126" Margin="353,508,0,0" VerticalAlignment="Top" Width="304"/>
 
    </Grid>
</Window>
"@


$inputXMLClean = $inputXML -replace 'mc:Ignorable="d"','' -replace "x:N",'N' -replace 'x:Class=".*?"','' -replace 'd:DesignHeight="\d*?"','' -replace 'd:DesignWidth="\d*?"',''
[xml]$xaml = $inputXMLClean
$reader = New-Object System.Xml.XmlNodeReader $xaml
$tempform = [Windows.Markup.XamlReader]::Load($reader)
$namedNodes = $xaml.SelectNodes("//*[@*[contains(translate(name(.),'n','N'),'Name')]]")
$namedNodes | ForEach-Object {$wpf.Add($_.Name, $tempform.FindName($_.Name))}

#Get the form name to be used as parameter in functions external to form...
$FormName = $NamedNodes[0].Name

#Cloud icon
#$Base64icon = "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"
#Exchange Management Shell icon
$Base64Icon = "iVBORw0KGgoAAAANSUhEUgAAABgAAAAYCAYAAADgdz34AAAAAXNSR0IArs4c6QAAAARnQU1BAACxjwv8YQUAAAAJcEhZcwAADsMAAA7DAcdvqGQAAAAYdEVYdFNvZnR3YXJlAHBhaW50Lm5ldCA0LjEuNBNAaMQAAAT/SURBVEhLnZV5TNNnGMfBYwOnmLA5TcTFRDz2xw4TDE6NDvHgCttUnM5MZIkynTcCRbEgyKmIZhDEDpkOSGBTBrI5dYCb8OtBixQo0BZb2tKWaltBBJSj3z1lYOSwjr3JNynh7ffzvs/zfZ/ajV4A7Ivq211ZRWrX5Wn1rk4sxnXpOZFb/G0NNmU3YFGcEC5RjPPQ9tcvDqN2TvxTFfwZpzF4Lrsq2InF27/352asv1iHBXEiOIQxsD/Ghf1RBnYhDN4IZeB0ZAKA5DuaQrfUGrwXK8S0cB4mkYk9yWo2nqwAu4kAEu9obBqO1v8EVI5rNp6GARn3tNodVxr1X10dqx2k038o9NnlSocRAPsQLqbQ5xlhXMxlC/Fh8n14fF8Lh1DeGEBQnswy+RiDSa/QVBLtcxwFYFCv68Tzfgt6+wfQNzCAZ339mBtVNQZQLjdLV1wQt71JIVgQK0K2QI8bEiPpEXw4Eky1Hmo8gNrcjfaePvxUZUBSWSsSSzWYGcEfA9hf0Bzsx5Fkr6EbXhe3IV/UhiyeHgpjFw4VKih9NgAtph6suFBLf1M8h4xHA3bnyS2L46tBEUfkb0rMYfMx66QAJfXG1wOMT3uRTCffnS/HslTxCMgwYFeuzLL8vBh7C2SYzRYM7pkXXYUymRkBV5qoBzYAFgvQ3TuAJ8/6cOqmarBpowGRJUrLuyd4mBXJR2COFNt+bIBrjBAHf5HD+QSf3hP31YBOMr4jNeFmgwl76Bb24wDCC5XzfS9JBo0cQyvBYXTI4urwDplbfWwC/u1B3QvTlzUM+LZAZlmWch9uKdX4hMroSOPkYoUGOVV6zKepMNkW4GFnLyJuKOGVKcFKSok178MAR4rfknX7jm7PUlnej2lCxDUtlidK4RIhxgen6nD5rgHbM2VwCRdhyaqvD4wCcKEyd6F/wAJT13O6STeqNU/oodFphgDzjtdjS3oPNqc/xaGcDmw9b4JPQhs2xrZiQ0wL/E8/gHe0FB60LyC+EWNu8MXlJrBKWpB2T4vCWiPuykwjAEtj1fA/34Odl7pQrXyKmAIj/BP08I7VwPuUAhvYUqw7UYe1rBocyJCPBFg1hcwcaKq+FcHDTErJ25Rt64Qd/B9pddIj+Kd24vPUDnBKH0Pa2ol9ma3wIfONUXIylwyafxoqRNYt/ViALU0Pq4LnGSP8Ujrgd8aEbal6lIofo1HVgS+TpNhwshGe1AuPMBE86R3wGtsnBphNX16fbITvWTN8kwzwidNiT7oKyrYuCOVGMq/F2vBqrCVzr3ABtlhTlPaXFtNZ3BdlsKWF7CZ4EcAvwQDfOB32ZGhRzDejVmFGWlEzndxqLoDHkUocTpfgSL7a0S5XoA9yiRYETQ3hBu3Kk5q86fEsShBhGo3sl3/drLn+KLoZPokP6eQ6HM8xgC/rwE2aogczGuDJuk/mQjKnPh38G5d/VyOfIcDLK4dv+DjkV4X7qtQa9xksrvu5Mm11WLESfj9IsDi+BitjlfCJ1+HSbRNaDN3ILVWBfVWO6xUGlPAf4hpVY2ecgAB3kZAr2xwFTBqyHn8x6nbnNL5hTmCBck5Ynirwm3QdxVGFvHI9kmjI+bLrEHS2AZySB8i51QxOsRQB7Er4RTDYFMldOGTz39Z3mW3+W1N02BitgBeVan0klSWcmhpKdQ/hUWkqsOZQOVYfKMfhNDGSciUEsLP7B/agGS/3eWk6AAAAAElFTkSuQmCC"

# Create a streaming image by streaming the base64 string to a bitmap streamsource
$bitmap = New-Object System.Windows.Media.Imaging.BitmapImage
$bitmap.BeginInit()
$bitmap.StreamSource = [System.IO.MemoryStream][System.Convert]::FromBase64String($Base64icon)
$bitmap.EndInit()
$bitmap.Freeze()
 
# This is the icon in the upper left hand corner of the app
$wpf.$FormName.Icon = $bitmap
#========================================================
# END of WPF form definition and load controls
#endregion
#========================================================

#========================================================
#region WPF EVENTS definition
#========================================================

#region Buttons
$wpf.btnRun.add_Click({
    Working-Label
    Get-Mailboxes
    if (($wpf.lblNbItemsInGrid.Content) -eq 0){
        $wpf.btnClipboard.IsEnabled = $false
    } Else {
        $wpf.btnClipboard.IsEnabled = $true
    }
    Ready-Label
})

$wpf.btnAction.add_Click({
    Working-Label
    Run-Action
    Ready-Label
})

$wpf.btnClipboard.add_Click({
    $CSVClip = $wpf.GridView.ItemsSource | ConvertTo-CSV -NoTypeInformation
    $CSVClip | clip.exe
    $title = "Copied !"
    $msg = "Data copied to the clipboard ! `n`rUse CTRL+V on Notepad or on Excel !"
    [System.Windows.MessageBox]::Show($msg,$title, "OK","Asterisk")
})

# End of Buttons region
#endregion

#region Load, Draw (render) and closing form events
#Things to load when the WPF form is loaded aka in memory
$Wpf.$FormName.Add_Loaded({
    Ready-Label
    Update-MainCommandLine
    $wpf.$FormName.Title += " - v$Version"
})
#Things to load when the WPF form is rendered aka drawn on screen
$Wpf.$FormName.Add_ContentRendered({
    write-host "rendered"
})

$Wpf.$FormName.add_Closing({
    $msg = "bye bye !"
    write-host $msg
})

$wpf.lblAbout.Add_MouseLeftButtonDown($lblabout_Click)

# End of load, draw and closing form events
#endregion

#region Text Changed events

$wpf.GridView.add_SelectionChanged({
    $Selected = $wpf.GridView.SelectedItems.count
    If ($Selected -eq 0) {
        $wpf.btnAction.IsEnabled = $false
        $wpf.comboSelectAction.IsEnabled = $false
    } ElseIf ($Selected -gt 0) {
        $wpf.btnAction.IsEnabled = $true
        $wpf.comboSelectAction.IsEnabled = $true
    }
    $wpf.lblNumberItemsSelected.Content = $Selected
})

$wpf.txtMailboxString.add_TextChanged({
    Update-MainCommandLine
})

$wpf.txtResultSize.add_TextChanged({
    Update-MainCommandLine
})

$wpf.chkUnlimited.add_Click({
    Update-MainCommandLine
    If ($wpf.chkUnlimited.IsChecked){
        $wpf.txtResultSize.IsEnabled = $false
    } Else {
        $wpf.txtResultSize.IsEnabled = $true
    }
})

$wpf.chkArbitrationOnly.add_Click({
    Update-MainCommandLine
    If ($Wpf.chkArbitrationOnly.IsChecked){
        $wpf.txtMailboxString.IsEnabled = $false
        $wpf.txtResultSize.IsEnabled = $false
        $wpf.chkUnlimited.IsEnabled = $false
    } Else {
        $wpf.txtMailboxString.IsEnabled = $true
        $wpf.txtResultSize.IsEnabled = $true
        $wpf.chkUnlimited.IsEnabled = $true
    }
})
#End of Text Changed events
#endregion


#endregion

#=======================================================
#End of Events from the WPF form
#endregion
#=======================================================

IsPSV3 | out-null

Test-ExchTools | out-null

# Load the form:
# Older way >>>>> $wpf.MyFormName.ShowDialog() | Out-Null >>>>> generates crash if run multiple times
# Newer way >>>>> avoiding crashes after a couple of launches in PowerShell...
# USing method from https://gist.github.com/altrive/6227237 to avoid crashing Powershell after we re-run the script after some inactivity time or if we run it several times consecutively...
$async = $wpf.$FormName.Dispatcher.InvokeAsync({
    $wpf.$FormName.ShowDialog() | Out-Null
})
$async.Wait() | Out-Null