Sunday, January 12, 2025

WPF GUI Using C# or PowerShell

C# and WPF (XAML) GUI


For displaying the output of Get-Process | Format-Table | Out-String in a WPF window, you have a few good options, each with its own strengths:

1. TextBox:

  • Pros: Simple to use, readily available. Handles scrolling automatically.
  • Cons: Limited formatting options. Might not preserve the table structure perfectly, especially if it involves complex formatting.

2. RichTextBox:

  • Pros: Offers more control over formatting. You can potentially retain some of the table-like structure by using tabs or other formatting.
  • Cons: Can be slightly more complex to work with than a TextBox.

3. DataGrid:

  • Pros: Excellent for displaying tabular data. If you can convert the output of Get-Process into objects, DataGrid will present them clearly in a structured grid.
  • Cons: Requires restructuring the PowerShell output into objects that the DataGrid can bind to.

4. ListView with GridView:

  • Pros: Provides a flexible way to display data in a grid-like format with customizable columns.
  • Cons: Like the DataGrid, it requires processing the output of Get-Process to fit the ListView structure.

Recommendation:

If you need to preserve the exact table formatting from Format-Table, RichTextBox might be the best choice. However, if you prioritize clear, structured presentation and are willing to convert the PowerShell output into objects, DataGrid is generally the most suitable option for displaying tabular data in WPF.

Example using DataGrid:

C#

using System.Collections.ObjectModel;
using System.Diagnostics;
using System.Management.Automation;
using System.Windows;

public partial class MainWindow : Window
{
   
public ObservableCollection<ProcessInfo> Processes { get; set; }

   
public MainWindow()
   {
       InitializeComponent();
       Processes =
new ObservableCollection<ProcessInfo>();
       DataContext =
this;
       LoadProcesses();
   }

   
private void LoadProcesses()
   {
       
using (PowerShell ps = PowerShell.Create())
       {
           ps.AddCommand(
"Get-Process");
           
foreach (PSObject result in ps.Invoke())
           {
               Processes.Add(
new ProcessInfo
               {
                   ProcessName = result.Properties[
"ProcessName"].Value.ToString(),
                   Id =
int.Parse(result.Properties["Id"].Value.ToString()),
                   
// Add other properties as needed
               });
           }
       }
   }
}

public class ProcessInfo
{
   
public string ProcessName { get; set; }
   
public int Id { get; set; }
   
// Add other properties as needed
}


XAML:

<Window ... >
   
<Grid>
       
<DataGrid ItemsSource="{Binding Processes}" AutoGenerateColumns="True" />
   
</Grid>
</Window>


This example demonstrates how to use DataGrid to display process information. You'll need to adjust the properties in ProcessInfo to match the ones you want to show from Get-Process.

PowerShell and XAML Script


So, the first example was using C# as a back-end language to invoke PowerShell and pass the “Get-Process” command via a string in ps.AddCommand property of the PowerShell object.  But, we can also use PowerShell as the language to load the XAML in a PowerShell session and iterate and create PSCustomObject’s for each invocation that finds each process one by one.  

Add-Type -AssemblyName PresentationFramework

# Create the XAML for the window
$xaml = @'
<Window
   xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
   xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
   Title="Process Viewer" Height="300" Width="500">
   <Grid>
       <DataGrid x:Name="dataGrid" AutoGenerateColumns="True" />
   </Grid>
</Window>
'@


# Create the WPF window

$window = [Windows.Markup.XamlReader]::Parse($xaml)

# Function to get process information
function Get-ProcessInfo {
   
Get-Process | ForEach-Object {
       [PSCustomObject]@{
           ProcessName =
$_.ProcessName
           Id           =
$_.Id
           
# Add other properties as needed
       }
   }
}

# Get process data
$processes = Get-ProcessInfo

# Bind the data to the DataGrid
$window.FindName("dataGrid").ItemsSource = $processes

# Show the window
$window.ShowDialog()

Step Details:

  1. Add WPF Assembly: Add-Type -AssemblyName PresentationFramework imports the necessary WPF libraries.
  2. Define XAML: The $xaml variable holds the XAML code for a simple window with a DataGrid.
  3. Create Window: [Windows.Markup.XamlReader]::Parse($xaml) creates the WPF window object from the XAML string.
  4. Get-ProcessInfo Function: This function fetches process information using Get-Process and formats it into custom objects with properties like ProcessName and Id. You can add more properties as needed.
  5. Bind Data: $window.FindName("dataGrid").ItemsSource = $processes finds the DataGrid in the window and sets its ItemsSource to the $processes collection, populating the grid.
  6. Show Window: $window.ShowDialog() displays the WPF window.

To run this code:

  1. Save it as a .ps1 file (e.g., process_viewer.ps1).
  2. Run the script from PowerShell: .\process_viewer.ps1

This will open a WPF window with a DataGrid displaying the process information.

This approach keeps the core logic in PowerShell, leveraging its capabilities for data retrieval and manipulation, while using WPF for the user interface.