# SPDX-License-Identifier: MIT # Set the maximum number of threads for the RunspacePool to the number of threads on the machine $maxthreads = [int]$env:NUMBER_OF_PROCESSORS # Create a new session state for parsing variables into our runspace $hashVars = New-object System.Management.Automation.Runspaces.SessionStateVariableEntry -ArgumentList 'sync',$sync,$Null $InitialSessionState = [System.Management.Automation.Runspaces.InitialSessionState]::CreateDefault() # Add the variable to the session state $InitialSessionState.Variables.Add($hashVars) # Get every private function and add them to the session state $functions = (Get-ChildItem function:\).where{$_.name -like "*winutil*" -or $_.name -like "*WPF*"} foreach ($function in $functions) { $functionDefinition = Get-Content function:\$($function.name) $functionEntry = New-Object System.Management.Automation.Runspaces.SessionStateFunctionEntry -ArgumentList $($function.name), $functionDefinition $initialSessionState.Commands.Add($functionEntry) } # Create the runspace pool $sync.runspace = [runspacefactory]::CreateRunspacePool( 1, # Minimum thread count $maxthreads, # Maximum thread count $InitialSessionState, # Initial session state $Host # Machine to create runspaces on ) # Open the RunspacePool instance $sync.runspace.Open() # Create classes for different exceptions class WingetFailedInstall : Exception { [string] $additionalData WingetFailedInstall($Message) : base($Message) {} } class ChocoFailedInstall : Exception { [string] $additionalData ChocoFailedInstall($Message) : base($Message) {} } class GenericException : Exception { [string] $additionalData GenericException($Message) : base($Message) {} } $inputXML = $inputXML -replace 'mc:Ignorable="d"', '' -replace "x:N", 'N' -replace '^ Set-WinUtilProgressbar $sync["Form"].TaskbarItemInfo = New-Object System.Windows.Shell.TaskbarItemInfo Set-WinUtilTaskbaritem -state "None" # Set the titlebar $sync["Form"].title = $sync["Form"].title + " " + $sync.version # Set the commands that will run when the form is closed $sync["Form"].Add_Closing({ $sync.runspace.Dispose() $sync.runspace.Close() [System.GC]::Collect() }) # Attach the event handler to the Click event $sync.SearchBarClearButton.Add_Click({ $sync.SearchBar.Text = "" $sync.SearchBarClearButton.Visibility = "Collapsed" }) # add some shortcuts for people that don't like clicking $commonKeyEvents = { if ($sync.ProcessRunning -eq $true) { return } if ($_.Key -eq "Escape") { $sync.SearchBar.SelectAll() $sync.SearchBar.Text = "" $sync.SearchBarClearButton.Visibility = "Collapsed" return } # don't ask, I know what I'm doing, just go... if (($_.Key -eq "Q" -and $_.KeyboardDevice.Modifiers -eq "Ctrl")) { $this.Close() } if ($_.KeyboardDevice.Modifiers -eq "Alt") { if ($_.SystemKey -eq "I") { Invoke-WPFButton "WPFTab1BT" } if ($_.SystemKey -eq "T") { Invoke-WPFButton "WPFTab2BT" } if ($_.SystemKey -eq "C") { Invoke-WPFButton "WPFTab3BT" } if ($_.SystemKey -eq "U") { Invoke-WPFButton "WPFTab4BT" } if ($_.SystemKey -eq "M") { Invoke-WPFButton "WPFTab5BT" } if ($_.SystemKey -eq "P") { Write-Host "Your Windows Product Key: $((Get-WmiObject -query 'select * from SoftwareLicensingService').OA3xOriginalProductKey)" } } # shortcut for the filter box if ($_.Key -eq "F" -and $_.KeyboardDevice.Modifiers -eq "Ctrl") { if ($sync.SearchBar.Text -eq "Ctrl-F to filter") { $sync.SearchBar.SelectAll() $sync.SearchBar.Text = "" } $sync.SearchBar.Focus() } } $sync["Form"].Add_PreViewKeyDown($commonKeyEvents) $sync["Form"].Add_MouseLeftButtonDown({ if ($sync["SettingsPopup"].IsOpen) { $sync["SettingsPopup"].IsOpen = $false } $sync["Form"].DragMove() }) $sync["Form"].Add_MouseDoubleClick({ if ($sync["Form"].WindowState -eq [Windows.WindowState]::Normal) { $sync["Form"].WindowState = [Windows.WindowState]::Maximized; } else { $sync["Form"].WindowState = [Windows.WindowState]::Normal; } }) $sync["Form"].Add_Deactivated({ Write-Debug "WinUtil lost focus" if ($sync["SettingsPopup"].IsOpen) { $sync["SettingsPopup"].IsOpen = $false } }) $sync["Form"].Add_ContentRendered({ try { [void][Window] } catch { Add-Type @" using System; using System.Runtime.InteropServices; public class Window { [DllImport("user32.dll")] public static extern IntPtr SendMessage(IntPtr hWnd, uint Msg, IntPtr wParam, IntPtr lParam); [DllImport("user32.dll")] [return: MarshalAs(UnmanagedType.Bool)] public static extern bool GetWindowRect(IntPtr hWnd, out RECT lpRect); [DllImport("user32.dll")] [return: MarshalAs(UnmanagedType.Bool)] public static extern bool MoveWindow(IntPtr handle, int x, int y, int width, int height, bool redraw); [DllImport("user32.dll")] public static extern int GetSystemMetrics(int nIndex); }; public struct RECT { public int Left; // x position of upper-left corner public int Top; // y position of upper-left corner public int Right; // x position of lower-right corner public int Bottom; // y position of lower-right corner } "@ } foreach ($proc in (Get-Process).where{ $_.MainWindowTitle -and $_.MainWindowTitle -like "*titus*" }) { # Check if the process's MainWindowHandle is valid if ($proc.MainWindowHandle -ne [System.IntPtr]::Zero) { Write-Debug "MainWindowHandle: $($proc.Id) $($proc.MainWindowTitle) $($proc.MainWindowHandle)" $windowHandle = $proc.MainWindowHandle } else { Write-Warning "Process found, but no MainWindowHandle: $($proc.Id) $($proc.MainWindowTitle)" } } $rect = New-Object RECT [Window]::GetWindowRect($windowHandle, [ref]$rect) $width = $rect.Right - $rect.Left $height = $rect.Bottom - $rect.Top Write-Debug "UpperLeft:$($rect.Left),$($rect.Top) LowerBottom:$($rect.Right),$($rect.Bottom). Width:$($width) Height:$($height)" # Load the Windows Forms assembly Add-Type -AssemblyName System.Windows.Forms $primaryScreen = [System.Windows.Forms.Screen]::PrimaryScreen # Check if the primary screen is found if ($primaryScreen) { # Extract screen width and height for the primary monitor $screenWidth = $primaryScreen.Bounds.Width $screenHeight = $primaryScreen.Bounds.Height # Print the screen size Write-Debug "Primary Monitor Width: $screenWidth pixels" Write-Debug "Primary Monitor Height: $screenHeight pixels" # Compare with the primary monitor size if ($width -gt $screenWidth -or $height -gt $screenHeight) { Write-Debug "The specified width and/or height is greater than the primary monitor size." [void][Window]::MoveWindow($windowHandle, 0, 0, $screenWidth, $screenHeight, $True) } else { Write-Debug "The specified width and height are within the primary monitor size limits." } } else { Write-Debug "Unable to retrieve information about the primary monitor." } Invoke-WPFTab "WPFTab1BT" $sync["Form"].Focus() # maybe this is not the best place to load and execute config file? # maybe community can help? if ($PARAM_CONFIG) { Invoke-WPFImpex -type "import" -Config $PARAM_CONFIG if ($PARAM_RUN) { while ($sync.ProcessRunning) { Start-Sleep -Seconds 5 } Start-Sleep -Seconds 5 Write-Host "Applying tweaks..." Invoke-WPFtweaksbutton while ($sync.ProcessRunning) { Start-Sleep -Seconds 5 } Start-Sleep -Seconds 5 Write-Host "Installing features..." Invoke-WPFFeatureInstall while ($sync.ProcessRunning) { Start-Sleep -Seconds 5 } Start-Sleep -Seconds 5 Write-Host "Installing applications..." while ($sync.ProcessRunning) { Start-Sleep -Seconds 1 } Invoke-WPFInstall Start-Sleep -Seconds 5 Write-Host "Done." } } }) $sync["ISORelease"].Items.Add("23H2") $sync["ISORelease"].Items.Add("22H2") $sync["ISORelease"].Items.Add("21H2") $sync["ISORelease"].SelectedItem = "23H2" $sync["ISOLanguage"].Items.Add((Get-Culture).Name) if ((Get-Culture).Name -ne "en-US") { $sync["ISOLanguage"].Items.Add("en-US") } $sync["ISOLanguage"].SelectedItem = (Get-Culture).Name $sync["ISOArchitecture"].Items.Add("64-bit") $sync["ISOArchitecture"].Items.Add("86-bit") $sync["ISOArchitecture"].SelectedItem = (Get-WmiObject -Class Win32_OperatingSystem).OSArchitecture if ($sync["ISOoption1"].IsChecked) { $sync["ISORelease"].Visibility = [System.Windows.Visibility]::Visible $sync["ISOLanguage"].Visibility = [System.Windows.Visibility]::Visible $sync["ISOArchitecture"].Visibility = [System.Windows.Visibility]::Visible } else { $sync["ISORelease"].Visibility = [System.Windows.Visibility]::Collapsed $sync["ISOLanguage"].Visibility = [System.Windows.Visibility]::Collapsed $sync["ISOArchitecture"].Visibility = [System.Windows.Visibility]::Collapsed } # Load Checkboxes and Labels outside of the Filter function only once on startup for performance reasons $filter = Get-WinUtilVariables -Type CheckBox $CheckBoxes = ($sync.GetEnumerator()).where{ $psitem.Key -in $filter } $filter = Get-WinUtilVariables -Type Label $labels = @{} ($sync.GetEnumerator()).where{$PSItem.Key -in $filter} | ForEach-Object {$labels[$_.Key] = $_.Value} $allCategories = $checkBoxes.Name | ForEach-Object {$sync.configs.applications.$_} | Select-Object -Unique -ExpandProperty category $sync["SearchBar"].Add_TextChanged({ if ($sync.SearchBar.Text -ne "") { $sync.SearchBarClearButton.Visibility = "Visible" } else { $sync.SearchBarClearButton.Visibility = "Collapsed" } $activeApplications = @() foreach ($CheckBox in $CheckBoxes) { # Check if the checkbox is null or if it doesn't have content if ($CheckBox -eq $null -or $CheckBox.Value -eq $null -or $CheckBox.Value.Content -eq $null) { continue } $textToSearch = $sync.SearchBar.Text.ToLower() $checkBoxName = $CheckBox.Key $textBlockName = $checkBoxName + "Link" # Retrieve the corresponding text block based on the generated name $textBlock = $sync[$textBlockName] if ($CheckBox.Value.Content.ToLower().Contains($textToSearch)) { $CheckBox.Value.Visibility = "Visible" $activeApplications += $sync.configs.applications.$checkboxName # Set the corresponding text block visibility if ($textBlock -ne $null) { $textBlock.Visibility = "Visible" } } else { $CheckBox.Value.Visibility = "Collapsed" # Set the corresponding text block visibility if ($textBlock -ne $null) { $textBlock.Visibility = "Collapsed" } } } $activeCategories = $activeApplications | Select-Object -ExpandProperty category -Unique foreach ($category in $activeCategories) { $label = $labels[$(Get-WPFObjectName -type "Label" -name $category)] $label.Visibility = "Visible" } if ($activeCategories) { $inactiveCategories = Compare-Object -ReferenceObject $allCategories -DifferenceObject $activeCategories -PassThru } else { $inactiveCategories = $allCategories } foreach ($category in $inactiveCategories) { $label = $labels[$(Get-WPFObjectName -type "Label" -name $category)] $label.Visibility = "Collapsed"} }) # Initialize the hashtable $winutildir = @{} # Set the path for the winutil directory $winutildir["path"] = "$env:LOCALAPPDATA\winutil\" if (-NOT (Test-Path -Path $winutildir["path"])) { New-Item -Path $winutildir["path"] -ItemType Directory } # Set the path for the logo and checkmark images $winutildir["logo.png"] = $winutildir["path"] + "cttlogo.png" $winutildir["logo.ico"] = $winutildir["path"] + "cttlogo.ico" if (-NOT (Test-Path -Path $winutildir["logo.png"])) { Invoke-WebRequest -Uri "https://christitus.com/images/logo-full.png" -OutFile $winutildir["logo.png"] } if (-NOT (Test-Path -Path $winutildir["logo.ico"])) { ConvertTo-Icon -bitmapPath $winutildir["logo.png"] -iconPath $winutildir["logo.ico"] } $winutildir["checkmark.png"] = $winutildir["path"] + "checkmark.png" $winutildir["warning.png"] = $winutildir["path"] + "warning.png" if (-NOT (Test-Path -Path $winutildir["checkmark.png"])) { Invoke-WebRequest -Uri "https://christitus.com/images/checkmark.png" -OutFile $winutildir["checkmark.png"] } if (-NOT (Test-Path -Path $winutildir["warning.png"])) { Invoke-WebRequest -Uri "https://christitus.com/images/warning.png" -OutFile $winutildir["warning.png"] } Set-WinUtilTaskbaritem -overlay "logo" $sync["Form"].Add_Activated({ Set-WinUtilTaskbaritem -overlay "logo" }) # Define event handler for button click $sync["SettingsButton"].Add_Click({ Write-Debug "SettingsButton clicked" if ($sync["SettingsPopup"].IsOpen) { $sync["SettingsPopup"].IsOpen = $false } else { $sync["SettingsPopup"].IsOpen = $true } $_.Handled = $false }) # Define event handlers for menu items $sync["ImportMenuItem"].Add_Click({ # Handle Import menu item click Write-Debug "Import clicked" $sync["SettingsPopup"].IsOpen = $false Invoke-WPFImpex -type "import" $_.Handled = $false }) $sync["ExportMenuItem"].Add_Click({ # Handle Export menu item click Write-Debug "Export clicked" $sync["SettingsPopup"].IsOpen = $false Invoke-WPFImpex -type "export" $_.Handled = $false }) $sync["AboutMenuItem"].Add_Click({ # Handle Export menu item click Write-Debug "About clicked" $sync["SettingsPopup"].IsOpen = $false $authorInfo = @" Author : @christitustech Runspace : @DeveloperDurp MicroWin : @KonTy GitHub : ChrisTitusTech/winutil Version : $($sync.version) "@ $FontSize = $sync.configs.themes.$ctttheme.CustomDialogFontSize $HeaderFontSize = $sync.configs.themes.$ctttheme.CustomDialogFontSizeHeader $IconSize = $sync.configs.themes.$ctttheme.CustomDialogIconSize $Width = $sync.configs.themes.$ctttheme.CustomDialogWidth $Height = $sync.configs.themes.$ctttheme.CustomDialogHeight Show-CustomDialog -Message $authorInfo -Width $Width -Height $Height -FontSize $FontSize -HeaderFontSize $HeaderFontSize -IconSize $IconSize }) $sync["SponsorMenuItem"].Add_Click({ # Handle Export menu item click Write-Debug "Sponsors clicked" $sync["SettingsPopup"].IsOpen = $false $authorInfo = @" Current sponsors for ChrisTitusTech: "@ $authorInfo += "`n" try { # Call the function to get the sponsors $sponsors = Invoke-WinUtilSponsors # Append the sponsors to the authorInfo $sponsors | ForEach-Object { $authorInfo += "$_`n" } } catch { $authorInfo += "An error occurred while fetching or processing the sponsors: $_`n" } $FontSize = $sync.configs.themes.$ctttheme.CustomDialogFontSize $HeaderFontSize = $sync.configs.themes.$ctttheme.CustomDialogFontSizeHeader $IconSize = $sync.configs.themes.$ctttheme.CustomDialogIconSize $Width = $sync.configs.themes.$ctttheme.CustomDialogWidth $Height = $sync.configs.themes.$ctttheme.CustomDialogHeight Show-CustomDialog -Message $authorInfo -Width $Width -Height $Height -FontSize $FontSize -HeaderFontSize $HeaderFontSize -IconSize $IconSize -EnableScroll $true }) $sync["Form"].ShowDialog() | out-null Stop-Transcript