Public/Start-RpRemotePro.ps1

function Start-RpRemotePro {
    <#
    .SYNOPSIS
    Initializes and manages the RemotePro application interface.
 
    .DESCRIPTION
    The Start-RpRemotePro function sets up a GUI for managing remote connections
    and commands. It uses WPF for dynamic display and runspaces for managing
    multiple asynchronous tasks, allowing for responsive interactions during
    complex operations. This function incorporates the MilestonePSTools and
    ImportExcel modules, providing essential tools for operation. It dynamically
    creates UI elements from provided XAML, handles user events, and executes
    commands. Extensive use of runspaces facilitates efficient execution of
    background PowerShell commands, managing long-running tasks, and handling
    concurrent operations to enhance performance.
 
    .PARAMETERS
    There are no parameters exposed in the function signature. However, the
    function uses internally managed variables and runtime configurations to
    control various features such as connection management, command execution,
    and event handling through the GUI.
 
    .COMPONENT
    RemoteProGUI
 
    .EXAMPLE
    Start-RpRemotePro
    # Launches the RemotePro GUI, initializes all components, and sets up
    runspaces for asynchronous task management.
 
    .NOTES
    - Requires that necessary modules and dependencies are installed and
    configured in the running environment.
    - Employs advanced features such as WPF and runspaces to maintain a smooth
    operation flow and responsive user interface during intensive tasks.
    - Runspaces are critical for the application's ability to perform multiple
    operations simultaneously, such as fetching data, updating the UI, and
    executing user commands without freezing or crashing.
 
    .LINK
    https://write-verbose.com/2023/03/21/PowerShellWPFPt1/
    Guide on integrating PowerShell with WPF for GUI applications.
 
    .LINK
    https://gist.github.com/proxb/6bc718831422df3392c4
    Example of managing complex PowerShell GUI applications.
 
    .LINK
    https://www.milestonepstools.com/
    MilestonePSTools website, detailing how it extends Milestone's MIP SDK capabilities
    to PowerShell for rapid VMS configuration and reporting automation.
 
    .LINK
    https://www.importexcel.com/
 
    .LINK
    https://github.com/EvotecIT/PSWriteHTML/blob/master/Docs/Out-HtmlView.md
 
    .LINK
    https://www.remotepro.dev/en-US/Start-RpRemotePro
    #>

    [CmdletBinding()]
    param()

    [System.GC]::Collect()

    # Ensure the RemotePro controller and runspace objects are initialized
    if (-not $script:RemotePro) {
        $script:RemotePro = New-RpControllerObject
    }
    if (-not $script:RpOpenRunspaces){
        $script:RpOpenRunspaces = Initialize-RpOpenRunspaces
    }
    if (-not $script:RpRunspaceJobs){
        $script:RpRunspaceJobs = Initialize-RpRunspaceJobs
    }
    if (-not $script:RpRunspaceResults){
        $script:RpRunspaceResults = Initialize-RpRunspaceResults
    }

    #region wpf xaml and reader
    # Modified XAML with an added button for opening the connection file
    $script:RemoteProXaml = $script:RemoteProXaml -replace '^<Window.*', '<Window' -replace 'mc:Ignorable="d"','' -replace "x:N",'N'
    [xml]$script:xaml = $script:RemoteProXaml

    $script:xmlreader = (New-Object System.Xml.XmlNodeReader $xaml)
    try {
        $script:window = [Windows.Markup.XamlReader]::Load($xmlreader)

        # Automatically find and create script-scoped variables for each named XAML element
        $script:xaml.SelectNodes("//*[@Name]") | ForEach-Object -Process {
            try {
                $variableName = $_.Name
                $variableValue = $script:window.FindName($variableName)

                if ($null -eq $variableValue) {
                    throw [System.InvalidOperationException] "No matching UI element found for '$variableName'."
                }

                # Note: since variables are set at the script scope "$script:" prefix is not needed but can be used.
                Set-Variable -Name $variableName -Value $variableValue -Scope Script

                # Write the variable name to the host
                Write-Verbose "Created script-scoped variable: $variableName"
            } catch {
                Write-Verbose "Failed to create script-scoped variable for: $($variableName): $($Error[0].Exception.Message)"
            }
        }
    } finally {
        $xmlreader.Close()
        $xmlreader.Dispose()
    }

    # Set the window icon
    if ($null -ne $script:window) {
        Set-RpWindowIcon -window $script:window
    } else {
        Write-Warning "WPF window failed to load. Cannot set icon."
    }

    #endregion

    #region vms connection and updating main window.
    Set-RpDefaultConnectionBox          # Helper function for clearing textbox
    Set-DefaultConnectionProfileBox     # Helper function for populating connection profile textbox.
    Get-RemoteProConnections            # Fill Connections_Combo_Box with profile names.
                                        # Refresh "MilestonePSTools Connection Profile Details" tab.
    Get-RpConnectionProfileRefresh      # Helper function to refresh connection profiles
                                        # Refresh profiles by default when the main window loads
    #endregion

    #region button-click processing
    # Attach event handlers to UI elements from RemotePro.EventHandlers
    # Syntax: $xamlVariableName.event($script:RemotePro.EventType.EventName)

    # Manage Connections TAB
    $script:New_Connection_File.Add_Click($script:RemotePro.EventHandlers.NewConnectionFile_Click)
    $script:OpenFile.Add_Click($script:RemotePro.EventHandlers.OpenFile_Click)
    $script:Connections_Combo_Box.Add_SelectionChanged($script:RemotePro.EventHandlers.ConnectionsComboBox_SelectionChanged)
    $script:ExecuteCommand.Add_Click($script:RemotePro.EventHandlers.ExecuteCommand_Click)
    $script:Connect.Add_Click($script:RemotePro.EventHandlers.Connect_Click)
    $script:Terminate.Add_Click($script:RemotePro.EventHandlers.Terminate_Click)
    $script:ConnectionProfileListBox.Add_SelectionChanged($script:RemotePro.EventHandlers.ConnectionProfileListBox_SelectionChanged)
    $script:PopOutButton.Add_Click($script:RemotePro.EventHandlers.PopOutButton_Click)
    $script:Connection_Profile_Refresh_Button.Add_Click($script:RemotePro.EventHandlers.Connection_Profile_Refresh_Button_Click)
    $script:Delete_Profile_Button.Add_Click($script:RemotePro.EventHandlers.DeleteProfileButton_Click)
    $script:Edit_Profile_Button.Add_Click($script:RemotePro.EventHandlers.EditProfileButton_Click)
    $script:Add_Profile_Button.Add_Click($script:RemotePro.EventHandlers.AddProfileButton_Click)
    $script:Connection_Status_Box.Add_TextChanged($script:RemotePro.EventHandlers.CSB_TextChanged)

    # Toolbar
    $script:GithubRepositoryCommand.Add_Click($script:RemotePro.EventHandlers.GithubRepositoryButton_Click)
    $script:PowerShellGalleryCommand.Add_Click($script:RemotePro.EventHandlers.PowerShellGalleryButton_Click)
    $script:DocsSiteCommand.Add_Click($script:RemotePro.EventHandlers.DocsSiteButton_Click)
    $script:FlowDirectionToggleButton.Add_Click($script:RemotePro.EventHandlers.FlowDirectionToggleButton_Click)
    $script:ReportIssueCommand.Add_Click($script:RemotePro.EventHandlers.ReportIssueButton_Click)
    $script:LicenseInformationCommand.Add_Click($script:RemotePro.EventHandlers.LicenseInformationButton_Click)
    $script:ExitApplicationCommand.Add_Click($script:RemotePro.EventHandlers.Window_AddClosed_Click)

    # Runspace log TAB
    $script:Runspace_Mutex_Log.Add_TextChanged($script:RemotePro.EventHandlers.RunspaceMutexLog_TextChanged)

    # Configuration & Settings TAB

    # Attach runspace events to UI elements from RemotePro.RunspaceEvents
    # Syntax: $xamlVariableName.event($script:RemotePro.EventType.RunspaceEventName)
    $script:CamReport.Add_Click($script:RemotePro.RunspaceEvents.CamReport_Click)
    $script:ShowCameras.Add_Click($script:RemotePro.RunspaceEvents.ShowCameras_Click)
    $script:TicketBlock.Add_Click($script:RemotePro.RunspaceEvents.TicketBlock_Click)
    $script:ShowVideoOSItems.Add_Click($script:RemotePro.RunspaceEvents.ShowVideoOSItems_Click)
    $script:Hardware.Add_Click($script:RemotePro.RunspaceEvents.Hardware_Click)
    $script:ItemState.Add_Click($script:RemotePro.RunspaceEvents.ItemState_Click)



    #endegion



    #region clean up main window environment
    $script:window.Add_Closed($script:RemotePro.EventHandlers.Window_AddClosed_Click)
    #endregion

    #region static runspace logic
    $jobsToRun = @(
        @{
            JobName                = "runspaceJob2"
            JobDetailsVariableName = "jobDetails2"
            Description            = "Get VmsCameraReport"
            ScriptBlock            = {}
        }
    )

    $script:RpOpenRunspaces = Start-RpRunspaceJobStatic -Jobs $jobsToRun -uiElement $script:Runspace_Mutex_Log -OpenRunspaces $script:RpOpenRunspaces -RunspaceJobs $script:RunspaceJobs -Verbose
    #$script:openRunspaces.Jobs | ? Runspace -eq $runspaceJob2 | select Runspace
    #$script:openRunspaces.Jobs | ogv
    #endregion

    #region runspace job timer, collection, and maintenance
    $script:RunspaceTimerParams = @{
        LogPath         = $script:logPath
        uiElement       = $script:Runspace_Mutex_Log
        RunspaceJobs    = $script:RunspaceJobs
        RunspaceResults = $script:RunspaceResults
        OpenRunspaces   = $script:RpOpenRunspaces
    }

    # Initialize the timer
    $script:RunspaceCleanupTimer = New-Object System.Windows.Forms.Timer
    $script:RunspaceCleanupTimer.Interval = 2000  # 2 seconds
    $script:RunspaceCleanupTimer.Add_Tick({
        Watch-RpRunspaces @script:RunspaceTimerParams
    })

    $script:RunspaceCleanupTimer.Start()
    #endregion

    $script:window.ShowDialog() | Out-Null

    if ($script:window){
        # Dispose of the window when closed
        $script:window.Close()
    }
}