You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
12893 lines
352 KiB
12893 lines
352 KiB
<#
|
|
.SYNOPSIS
|
|
Sophia Script is a PowerShell module for Windows 10 & Windows 11 fine-tuning and automating the routine tasks
|
|
|
|
Version: v6.1.2
|
|
Date: 16.07.2022
|
|
|
|
Copyright (c) 2014—2022 farag
|
|
Copyright (c) 2019—2022 farag & Inestic
|
|
|
|
Thanks to all https://forum.ru-board.com members involved
|
|
|
|
.NOTES
|
|
Running the script is best done on a fresh install because running it on wrong tweaked system may result in errors occurring
|
|
|
|
.NOTES
|
|
Supported Windows 11 versions
|
|
Versions: 21H2/22H2
|
|
Builds: 22000.739+, 22509+
|
|
Editions: Home/Pro/Enterprise
|
|
|
|
.NOTES
|
|
Set execution policy to be able to run scripts only in the current PowerShell session:
|
|
Set-ExecutionPolicy -ExecutionPolicy Bypass -Scope Process -Force
|
|
|
|
.LINK GitHub
|
|
https://github.com/farag2/Sophia-Script-for-Windows
|
|
|
|
.LINK Telegram channel & group
|
|
https://t.me/sophianews
|
|
https://t.me/sophia_chat
|
|
|
|
.NOTES
|
|
https://forum.ru-board.com/topic.cgi?forum=62&topic=30617#15
|
|
https://habr.com/company/skillfactory/blog/553800/
|
|
https://forums.mydigitallife.net/threads/powershell-windows-10-sophia-script.81675/
|
|
https://www.reddit.com/r/PowerShell/comments/go2n5v/powershell_script_setup_windows_10/
|
|
|
|
.LINK Authors
|
|
https://github.com/farag2
|
|
https://github.com/Inestic
|
|
#>
|
|
|
|
#region Checkings
|
|
function Checkings
|
|
{
|
|
param
|
|
(
|
|
[Parameter(Mandatory = $false)]
|
|
[switch]
|
|
$Warning
|
|
)
|
|
|
|
Set-StrictMode -Version Latest
|
|
|
|
# Сlear the $Error variable
|
|
$Global:Error.Clear()
|
|
|
|
# Detect the OS build version
|
|
switch ((Get-CimInstance -ClassName Win32_OperatingSystem).BuildNumber -ge 22000)
|
|
{
|
|
$false
|
|
{
|
|
Write-Warning -Message $Localization.UnsupportedOSBuild
|
|
|
|
# Enable receiving updates for other Microsoft products when you update Windows
|
|
(New-Object -ComObject Microsoft.Update.ServiceManager).AddService2("7971f918-a847-4430-9279-4a52d1efe18d", 7, "")
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Check for UWP apps updates
|
|
Get-CimInstance -Namespace "Root\cimv2\mdm\dmmap" -ClassName "MDM_EnterpriseModernAppManagement_AppManagement01" | Invoke-CimMethod -MethodName UpdateScanMethod
|
|
|
|
# Open the "Windows Update" page
|
|
Start-Process -FilePath "ms-settings:windowsupdate-action"
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Trigger Windows Update for detecting new updates
|
|
(New-Object -ComObject Microsoft.Update.AutoUpdate).DetectNow()
|
|
|
|
Start-Process -FilePath "https://t.me/sophia_chat"
|
|
|
|
exit
|
|
}
|
|
}
|
|
|
|
# Check whether the OS minor build version is 739 minimum
|
|
# https://docs.microsoft.com/en-us/windows/release-health/windows11-release-information
|
|
# https://support.microsoft.com/en-us/topic/june-14-2022-kb5014697-os-build-22000-739-cd3aaa0b-a8da-44a0-a778-dfb6f1d9ea11
|
|
switch ((Get-ItemPropertyValue -Path "HKLM:\SOFTWARE\Microsoft\Windows nt\CurrentVersion" -Name UBR) -ge 739)
|
|
{
|
|
$false
|
|
{
|
|
if ((Get-CimInstance -ClassName Win32_OperatingSystem).BuildNumber -eq 22000)
|
|
{
|
|
$Version = Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows nt\CurrentVersion"
|
|
Write-Warning -Message ($Localization.UpdateWarning -f $Version.CurrentBuild, $Version.UBR)
|
|
|
|
# Enable receiving updates for other Microsoft products when you update Windows
|
|
(New-Object -ComObject Microsoft.Update.ServiceManager).AddService2("7971f918-a847-4430-9279-4a52d1efe18d", 7, "")
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Check for UWP apps updates
|
|
Get-CimInstance -Namespace "Root\cimv2\mdm\dmmap" -ClassName "MDM_EnterpriseModernAppManagement_AppManagement01" | Invoke-CimMethod -MethodName UpdateScanMethod
|
|
|
|
# Open the "Windows Update" page
|
|
Start-Process -FilePath "ms-settings:windowsupdate-action"
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Trigger Windows Update for detecting new updates
|
|
(New-Object -ComObject Microsoft.Update.AutoUpdate).DetectNow()
|
|
|
|
Start-Process -FilePath "https://t.me/sophia_chat"
|
|
|
|
exit
|
|
}
|
|
}
|
|
}
|
|
|
|
# Check the language mode
|
|
switch ($ExecutionContext.SessionState.LanguageMode -ne "FullLanguage")
|
|
{
|
|
$true
|
|
{
|
|
Write-Warning -Message $Localization.UnsupportedLanguageMode
|
|
|
|
Start-Process -FilePath "https://docs.microsoft.com/en-us/powershell/module/microsoft.powershell.core/about/about_language_modes"
|
|
Start-Process -FilePath "https://t.me/sophia_chat"
|
|
|
|
exit
|
|
}
|
|
}
|
|
|
|
# Check whether the logged-in user is an admin
|
|
$CurrentUserName = (Get-Process -Id $PID -IncludeUserName).UserName | Split-Path -Leaf
|
|
$CurrentSessionId = (Get-Process -Id $PID -IncludeUserName).SessionId
|
|
$LoginUserName = (Get-Process -IncludeUserName | Where-Object -FilterScript {($_.ProcessName -eq "explorer") -and ($_.SessionId -eq $CurrentSessionId)}).UserName | Select-Object -First 1 | Split-Path -Leaf
|
|
|
|
switch ($CurrentUserName -ne $LoginUserName)
|
|
{
|
|
$true
|
|
{
|
|
Write-Warning -Message $Localization.LoggedInUserNotAdmin
|
|
Start-Process -FilePath "https://t.me/sophia_chat"
|
|
exit
|
|
}
|
|
}
|
|
|
|
# Check whether the script was run via PowerShell 5.1
|
|
if ($PSVersionTable.PSVersion.Major -ne 5)
|
|
{
|
|
Write-Warning -Message ($Localization.UnsupportedPowerShell -f $PSVersionTable.PSVersion.Major, $PSVersionTable.PSVersion.Minor)
|
|
Start-Process -FilePath "https://t.me/sophia_chat"
|
|
exit
|
|
}
|
|
|
|
# Check whether the script was run via PowerShell ISE
|
|
if ($Host.Name -match "ISE")
|
|
{
|
|
Write-Warning -Message $Localization.UnsupportedISE
|
|
Start-Process -FilePath "https://t.me/sophia_chat"
|
|
exit
|
|
}
|
|
|
|
# Check whether the OS was infected by the Win 10 Tweaker's trojan
|
|
# https://win10tweaker.ru
|
|
if (Test-Path -Path "HKCU:\Software\Win 10 Tweaker")
|
|
{
|
|
Write-Warning -Message $Localization.Win10TweakerWarning
|
|
|
|
Start-Process -FilePath "https://youtu.be/na93MS-1EkM"
|
|
Start-Process -FilePath "https://pikabu.ru/story/byekdor_v_win_10_tweaker_ili_sovremennyie_metodyi_borbyi_s_piratstvom_8227558"
|
|
Start-Process -FilePath "https://t.me/sophia_chat"
|
|
|
|
exit
|
|
}
|
|
|
|
# Check whether the OS was destroyed by Sycnex's Windows10Debloater script
|
|
# https://github.com/Sycnex/Windows10Debloater
|
|
if (Test-Path -Path $env:SystemDrive\Temp\Windows10Debloater)
|
|
{
|
|
Write-Warning -Message $Localization.Windows10DebloaterWarning
|
|
exit
|
|
}
|
|
|
|
# Check whether libraries exist in the bin folder
|
|
$Libraries = @(
|
|
"$PSScriptRoot\..\bin\Start_Layout\start.bin",
|
|
"$PSScriptRoot\..\bin\PolicyFileEditor\Commands.ps1",
|
|
"$PSScriptRoot\..\bin\PolicyFileEditor\Common.ps1",
|
|
"$PSScriptRoot\..\bin\PolicyFileEditor\PolFileEditor.dll",
|
|
"$PSScriptRoot\..\bin\PolicyFileEditor\PolicyFileEditor.psd1",
|
|
"$PSScriptRoot\..\bin\PolicyFileEditor\PolicyFileEditor.psm1"
|
|
)
|
|
if (($Libraries | Test-Path) -contains $false)
|
|
{
|
|
Write-Warning -Message $Localization.Bin
|
|
|
|
Start-Sleep -Seconds 5
|
|
|
|
Start-Process -FilePath "https://github.com/farag2/Sophia-Script-for-Windows/releases/latest"
|
|
Start-Process -FilePath "https://t.me/sophia_chat"
|
|
|
|
exit
|
|
}
|
|
|
|
# Check for a pending reboot
|
|
$PendingActions = @(
|
|
# CBS pending
|
|
"HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Component Based Servicing\RebootPending",
|
|
"HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Component Based Servicing\RebootInProgress",
|
|
"HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Component Based Servicing\PackagesPending",
|
|
# Windows Update pending
|
|
"HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\WindowsUpdate\Auto Update\PostRebootReporting",
|
|
"HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\WindowsUpdate\Auto Update\RebootRequired"
|
|
)
|
|
if (($PendingActions | Test-Path) -contains $true)
|
|
{
|
|
Write-Warning -Message $Localization.RebootPending
|
|
exit
|
|
}
|
|
|
|
# Check if the current module version is the latest one
|
|
try
|
|
{
|
|
[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12
|
|
|
|
# Check the internet connection
|
|
$Parameters = @{
|
|
Uri = "https://www.google.com"
|
|
Method = "Head"
|
|
DisableKeepAlive = $true
|
|
UseBasicParsing = $true
|
|
}
|
|
if (-not (Invoke-WebRequest @Parameters).StatusDescription)
|
|
{
|
|
return
|
|
}
|
|
|
|
try
|
|
{
|
|
# https://github.com/farag2/Sophia-Script-for-Windows/blob/master/sophia_script_versions.json
|
|
$Parameters = @{
|
|
Uri = "https://raw.githubusercontent.com/farag2/Sophia-Script-for-Windows/master/sophia_script_versions.json"
|
|
UseBasicParsing = $true
|
|
}
|
|
$LatestRelease = (Invoke-RestMethod @Parameters).Sophia_Script_Windows_11_PowerShell_5_1
|
|
$CurrentRelease = (Get-Module -Name Sophia).Version.ToString()
|
|
switch ([System.Version]$LatestRelease -gt [System.Version]$CurrentRelease)
|
|
{
|
|
$true
|
|
{
|
|
Write-Warning -Message $Localization.UnsupportedRelease
|
|
|
|
Start-Sleep -Seconds 5
|
|
|
|
Start-Process -FilePath "https://github.com/farag2/Sophia-Script-for-Windows/releases/latest"
|
|
Start-Process -FilePath "https://t.me/sophia_chat"
|
|
exit
|
|
}
|
|
}
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message ($Localization.NoResponse -f "https://github.com")
|
|
Write-Error -Message ($Localization.NoResponse -f "https://github.com") -ErrorAction SilentlyContinue
|
|
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message $Localization.NoInternetConnection
|
|
Write-Error -Message $Localization.NoInternetConnection -ErrorAction SilentlyContinue
|
|
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
|
|
# Unblock all files in the script folder by removing the Zone.Identifier alternate data stream with a value of "3"
|
|
Get-ChildItem -Path $PSScriptRoot\..\ -File -Recurse -Force | Unblock-File
|
|
|
|
#region Defender Checkings
|
|
# Checking whether WMI is corrupted
|
|
try
|
|
{
|
|
Get-CimInstance -ClassName MSFT_MpComputerStatus -Namespace root/microsoft/windows/defender -ErrorAction Stop | Out-Null
|
|
}
|
|
catch [Microsoft.Management.Infrastructure.CimException]
|
|
{
|
|
# Provider Load Failure exception
|
|
$Global:Error.Exception[-1]
|
|
Start-Process -FilePath "https://t.me/sophia_chat"
|
|
|
|
exit
|
|
}
|
|
|
|
# Check Microsoft Defender state
|
|
$productState = (Get-CimInstance -Namespace root/SecurityCenter2 -ClassName Antivirusproduct | Where-Object -FilterScript {$_.instanceGuid -eq "{D68DDC3A-831F-4fae-9E44-DA132C1ACF46}"}).productState
|
|
$DefenderState = ('0x{0:x}' -f $productState).Substring(3, 2)
|
|
if ($DefenderState -notmatch "00|01")
|
|
{
|
|
$Script:DefenderproductState = $true
|
|
}
|
|
else
|
|
{
|
|
$Script:DefenderproductState = $false
|
|
}
|
|
|
|
# Checking services
|
|
@("Windefend", "SecurityHealthService", "wscsvc") | ForEach-Object -Process {
|
|
if ($null -eq (Get-Service -Name $_ -ErrorAction Ignore))
|
|
{
|
|
$Localization.DefenderBroken
|
|
exit
|
|
}
|
|
else
|
|
{
|
|
if ((Get-Service -Name $_ -ErrorAction Ignore).Status -eq "running")
|
|
{
|
|
$Script:DefenderServices = $true
|
|
}
|
|
else
|
|
{
|
|
$Script:DefenderServices = $false
|
|
}
|
|
}
|
|
}
|
|
|
|
if ((Get-CimInstance -ClassName MSFT_MpComputerStatus -Namespace root/microsoft/windows/defender).AntispywareEnabled)
|
|
{
|
|
$Script:DefenderAntispywareEnabled = $true
|
|
}
|
|
else
|
|
{
|
|
$Script:DefenderAntispywareEnabled = $false
|
|
}
|
|
|
|
# https://docs.microsoft.com/en-us/graph/api/resources/intune-devices-windowsdefenderproductstatus?view=graph-rest-beta
|
|
if ((Get-CimInstance -ClassName MSFT_MpComputerStatus -Namespace root/microsoft/windows/defender).ProductStatus -eq 1)
|
|
{
|
|
$Script:DefenderProductStatus = $false
|
|
}
|
|
else
|
|
{
|
|
$Script:DefenderProductStatus = $true
|
|
}
|
|
|
|
# https://docs.microsoft.com/en-us/graph/api/resources/intune-devices-windowsdefenderproductstatus?view=graph-rest-beta
|
|
if ((Get-CimInstance -ClassName MSFT_MpComputerStatus -Namespace root/microsoft/windows/defender).AMEngineVersion -eq "0.0.0.0")
|
|
{
|
|
$Script:DefenderAMEngineVersion = $false
|
|
}
|
|
else
|
|
{
|
|
$Script:DefenderAMEngineVersion = $true
|
|
}
|
|
|
|
# Check whether Microsoft Defender was turned off
|
|
# Due to "Set-StrictMode -Version Latest" we have to use try/catch & GetValue()
|
|
try
|
|
{
|
|
if ([Microsoft.Win32.Registry]::GetValue("HKEY_LOCAL_MACHINE\SOFTWARE\Policies\Microsoft\Windows Defender", "DisableAntiSpyware", $false) -eq 1)
|
|
{
|
|
$Script:DisableAntiSpyware = $true
|
|
}
|
|
else
|
|
{
|
|
$Script:DisableAntiSpyware = $false
|
|
}
|
|
}
|
|
catch {}
|
|
|
|
# Check whether real-time protection prompts for known malware detection
|
|
# Due to "Set-StrictMode -Version Latest" we have to use try/catch & GetValue()
|
|
try
|
|
{
|
|
if ([Microsoft.Win32.Registry]::GetValue("HKEY_LOCAL_MACHINE\SOFTWARE\Policies\Microsoft\Windows Defender\Real-Time Protection", "DisableRealtimeMonitoring", $false) -eq 1)
|
|
{
|
|
$Script:DisableRealtimeMonitoring = $true
|
|
}
|
|
else
|
|
{
|
|
$Script:DisableRealtimeMonitoring = $false
|
|
}
|
|
}
|
|
catch {}
|
|
|
|
# Check whether behavior monitoring was disabled
|
|
# Due to "Set-StrictMode -Version Latest" we have to use try/catch & GetValue()
|
|
try
|
|
{
|
|
if ([Microsoft.Win32.Registry]::GetValue("HKEY_LOCAL_MACHINE\SOFTWARE\Policies\Microsoft\Windows Defender\Real-Time Protection", "DisableBehaviorMonitoring", $false) -eq 1)
|
|
{
|
|
$Script:DisableBehaviorMonitoring = $true
|
|
}
|
|
else
|
|
{
|
|
$Script:DisableBehaviorMonitoring = $false
|
|
}
|
|
}
|
|
catch {}
|
|
|
|
if ($Script:DefenderproductState -and $Script:DefenderServices -and $Script:DefenderAntispywareEnabled -and $Script:DefenderProductStatus -and
|
|
$Script:DefenderAMEngineVersion -and (-not $Script:DisableAntiSpyware) -and (-not $Script:DisableRealtimeMonitoring) -and (-not $Script:DisableBehaviorMonitoring))
|
|
{
|
|
# Defender is enabled
|
|
$Script:DefenderEnabled = $true
|
|
|
|
switch ((Get-MpPreference).EnableControlledFolderAccess)
|
|
{
|
|
"1"
|
|
{
|
|
Write-Warning -Message $Localization.ControlledFolderAccessDisabled
|
|
|
|
# Turn off Controlled folder access to let the script proceed
|
|
$Script:ControlledFolderAccess = $true
|
|
Set-MpPreference -EnableControlledFolderAccess Disabled
|
|
|
|
# Open "Ransomware protection" page
|
|
Start-Process -FilePath windowsdefender://RansomwareProtection
|
|
}
|
|
"0"
|
|
{
|
|
$Script:ControlledFolderAccess = $false
|
|
}
|
|
}
|
|
}
|
|
#endregion Defender Checkings
|
|
|
|
# Display a warning message about whether a user has customized the preset file
|
|
if ($Warning)
|
|
{
|
|
# Get the name of a preset (e.g Sophia.ps1) regardless it was named
|
|
$PresetName = Split-Path -Path ((Get-PSCallStack).Position | Where-Object -FilterScript {$_.File -match ".ps1"}).File -Leaf
|
|
|
|
$Title = ""
|
|
$Message = $Localization.CustomizationWarning -f $PresetName
|
|
$Yes = $Localization.Yes
|
|
$No = $Localization.No
|
|
$Options = "&$No", "&$Yes"
|
|
$DefaultChoice = 0
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
Invoke-Item -Path $PSScriptRoot\..\$PresetName
|
|
|
|
Start-Sleep -Seconds 5
|
|
|
|
Start-Process -FilePath "https://github.com/farag2/Sophia-Script-for-Windows#how-to-use"
|
|
exit
|
|
}
|
|
"1"
|
|
{
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
|
|
# Save all opened folders in order to restore them after File Explorer restart
|
|
$Script:OpenedFolders = {(New-Object -ComObject Shell.Application).Windows() | ForEach-Object -Process {$_.Document.Folder.Self.Path}}.Invoke()
|
|
}
|
|
#endregion Checkings
|
|
|
|
#region Protection
|
|
# Enable script logging. The log will be being recorded into the script root folder
|
|
# To stop logging just close the console or type "Stop-Transcript"
|
|
function Logging
|
|
{
|
|
$TrascriptFilename = "Log-$((Get-Date).ToString("dd.MM.yyyy-HH-mm"))"
|
|
Start-Transcript -Path $PSScriptRoot\..\$TrascriptFilename.txt -Force
|
|
}
|
|
|
|
# Create a restore point for the system drive
|
|
function CreateRestorePoint
|
|
{
|
|
$SystemDriveUniqueID = (Get-Volume | Where-Object -FilterScript {$_.DriveLetter -eq "$($env:SystemDrive[0])"}).UniqueID
|
|
$SystemProtection = ((Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\SPP\Clients" -ErrorAction Ignore)."{09F7EDC5-294E-4180-AF6A-FB0E6A0E9513}") | Where-Object -FilterScript {$_ -match [regex]::Escape($SystemDriveUniqueID)}
|
|
|
|
$ComputerRestorePoint = $false
|
|
|
|
switch ($null -eq $SystemProtection)
|
|
{
|
|
$true
|
|
{
|
|
$ComputerRestorePoint = $true
|
|
Enable-ComputerRestore -Drive $env:SystemDrive
|
|
}
|
|
}
|
|
|
|
# Never skip creating a restore point
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\SystemRestore" -Name SystemRestorePointCreationFrequency -PropertyType DWord -Value 0 -Force
|
|
|
|
Checkpoint-Computer -Description "Sophia Script for Windows 11" -RestorePointType MODIFY_SETTINGS
|
|
|
|
# Revert the System Restore checkpoint creation frequency to 1440 minutes
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\SystemRestore" -Name SystemRestorePointCreationFrequency -PropertyType DWord -Value 1440 -Force
|
|
|
|
# Turn off System Protection for the system drive if it was turned off before without deleting the existing restore points
|
|
if ($ComputerRestorePoint)
|
|
{
|
|
Disable-ComputerRestore -Drive $env:SystemDrive
|
|
}
|
|
}
|
|
#endregion Protection
|
|
|
|
#region Privacy & Telemetry
|
|
<#
|
|
.SYNOPSIS
|
|
The Connected User Experiences and Telemetry (DiagTrack) service
|
|
|
|
.PARAMETER Disable
|
|
Disable the Connected User Experiences and Telemetry (DiagTrack) service, and block connection for the Unified Telemetry Client Outbound Traffic
|
|
|
|
.PARAMETER Enable
|
|
Enable the Connected User Experiences and Telemetry (DiagTrack) service, and allow connection for the Unified Telemetry Client Outbound Traffic
|
|
|
|
.EXAMPLE
|
|
DiagTrackService -Disable
|
|
|
|
.EXAMPLE
|
|
DiagTrackService -Enable
|
|
|
|
.NOTES
|
|
Disabling the "Connected User Experiences and Telemetry" service (DiagTrack) can cause you not being able to get Xbox achievements anymore
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function DiagTrackService
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
# Connected User Experiences and Telemetry
|
|
# Disabling the "Connected User Experiences and Telemetry" service (DiagTrack) can cause you not being able to get Xbox achievements anymore
|
|
Get-Service -Name DiagTrack | Stop-Service -Force
|
|
Get-Service -Name DiagTrack | Set-Service -StartupType Disabled
|
|
|
|
# Block connection for the Unified Telemetry Client Outbound Traffic
|
|
Get-NetFirewallRule -Group DiagTrack | Set-NetFirewallRule -Enabled False -Action Block
|
|
}
|
|
"Enable"
|
|
{
|
|
# Connected User Experiences and Telemetry
|
|
Get-Service -Name DiagTrack | Set-Service -StartupType Automatic
|
|
Get-Service -Name DiagTrack | Start-Service
|
|
|
|
# Allow connection for the Unified Telemetry Client Outbound Traffic
|
|
Get-NetFirewallRule -Group DiagTrack | Set-NetFirewallRule -Enabled True -Action Allow
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Diagnostic data
|
|
|
|
.PARAMETER Minimal
|
|
Set the diagnostic data collection to minimum
|
|
|
|
.PARAMETER Default
|
|
Set the diagnostic data collection to default
|
|
|
|
.EXAMPLE
|
|
DiagnosticDataLevel -Minimal
|
|
|
|
.EXAMPLE
|
|
DiagnosticDataLevel -Default
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function DiagnosticDataLevel
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Minimal"
|
|
)]
|
|
[switch]
|
|
$Minimal,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Default"
|
|
)]
|
|
[switch]
|
|
$Default
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Minimal"
|
|
{
|
|
if (Get-WindowsEdition -Online | Where-Object -FilterScript {($_.Edition -like "Enterprise*") -or ($_.Edition -eq "Education")})
|
|
{
|
|
# Diagnostic data off
|
|
if (-not (Test-Path -Path HKLM:\Software\Policies\Microsoft\Windows\DataCollection))
|
|
{
|
|
New-Item -Path HKLM:\Software\Policies\Microsoft\Windows\DataCollection -Force
|
|
}
|
|
New-ItemProperty -Path HKLM:\Software\Policies\Microsoft\Windows\DataCollection -Name AllowTelemetry -PropertyType DWord -Value 0 -Force
|
|
}
|
|
else
|
|
{
|
|
# Send required diagnostic data
|
|
if (-not (Test-Path -Path HKLM:\Software\Policies\Microsoft\Windows\DataCollection))
|
|
{
|
|
New-Item -Path HKLM:\Software\Policies\Microsoft\Windows\DataCollection -Force
|
|
}
|
|
New-ItemProperty -Path HKLM:\Software\Policies\Microsoft\Windows\DataCollection -Name AllowTelemetry -PropertyType DWord -Value 1 -Force
|
|
}
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\DataCollection -Name MaxTelemetryAllowed -PropertyType DWord -Value 1 -Force
|
|
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Diagnostics\DiagTrack -Name ShowedToastAtLevel -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Default"
|
|
{
|
|
# Optional diagnostic data
|
|
Remove-ItemProperty -Path HKLM:\Software\Policies\Microsoft\Windows\DataCollection -Name AllowTelemetry -Force
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\DataCollection -Name MaxTelemetryAllowed -PropertyType DWord -Value 3 -Force
|
|
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Diagnostics\DiagTrack -Name ShowedToastAtLevel -PropertyType DWord -Value 3 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Windows Error Reporting
|
|
|
|
.PARAMETER Disable
|
|
Turn off Windows Error Reporting
|
|
|
|
.PARAMETER Enable
|
|
Turn on Windows Error Reporting
|
|
|
|
.EXAMPLE
|
|
ErrorReporting -Disable
|
|
|
|
.EXAMPLE
|
|
ErrorReporting -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function ErrorReporting
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if ((Get-WindowsEdition -Online).Edition -notmatch "Core")
|
|
{
|
|
Get-ScheduledTask -TaskName QueueReporting | Disable-ScheduledTask
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\Windows Error Reporting" -Name Disabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
|
|
Get-Service -Name WerSvc | Stop-Service -Force
|
|
Get-Service -Name WerSvc | Set-Service -StartupType Disabled
|
|
}
|
|
"Enable"
|
|
{
|
|
Get-ScheduledTask -TaskName QueueReporting | Enable-ScheduledTask
|
|
Remove-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\Windows Error Reporting" -Name Disabled -Force -ErrorAction Ignore
|
|
|
|
Get-Service -Name WerSvc | Set-Service -StartupType Manual
|
|
Get-Service -Name WerSvc | Start-Service
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The feedback frequency
|
|
|
|
.PARAMETER Never
|
|
Change the feedback frequency to "Never"
|
|
|
|
.PARAMETER Automatically
|
|
Change feedback frequency to "Automatically"
|
|
|
|
.EXAMPLE
|
|
FeedbackFrequency -Never
|
|
|
|
.EXAMPLE
|
|
FeedbackFrequency -Automatically
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function FeedbackFrequency
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Never"
|
|
)]
|
|
[switch]
|
|
$Never,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Automatically"
|
|
)]
|
|
[switch]
|
|
$Automatically
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Never"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Siuf\Rules))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Siuf\Rules -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Siuf\Rules -Name NumberOfSIUFInPeriod -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Automatically"
|
|
{
|
|
Remove-Item -Path HKCU:\SOFTWARE\Microsoft\Siuf\Rules -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The diagnostics tracking scheduled tasks
|
|
|
|
.PARAMETER Disable
|
|
Turn off the diagnostics tracking scheduled tasks
|
|
|
|
.PARAMETER Enable
|
|
Turn on the diagnostics tracking scheduled tasks
|
|
|
|
.EXAMPLE
|
|
ScheduledTasks -Disable
|
|
|
|
.EXAMPLE
|
|
ScheduledTasks -Enable
|
|
|
|
.NOTES
|
|
A pop-up dialog box lets a user select tasks
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function ScheduledTasks
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
Add-Type -AssemblyName PresentationCore, PresentationFramework
|
|
|
|
#region Variables
|
|
# Initialize an array list to store the selected scheduled tasks
|
|
$SelectedTasks = New-Object -TypeName System.Collections.ArrayList($null)
|
|
|
|
# The following tasks will have their checkboxes checked
|
|
[string[]]$CheckedScheduledTasks = @(
|
|
# Collects program telemetry information if opted-in to the Microsoft Customer Experience Improvement Program
|
|
"ProgramDataUpdater",
|
|
|
|
# This task collects and uploads autochk SQM data if opted-in to the Microsoft Customer Experience Improvement Program
|
|
"Proxy",
|
|
|
|
# If the user has consented to participate in the Windows Customer Experience Improvement Program, this job collects and sends usage data to Microsoft
|
|
"Consolidator",
|
|
|
|
# The USB CEIP (Customer Experience Improvement Program) task collects Universal Serial Bus related statistics and information about your machine and sends it to the Windows Device Connectivity engineering group at Microsoft
|
|
"UsbCeip",
|
|
|
|
# The Windows Disk Diagnostic reports general disk and system information to Microsoft for users participating in the Customer Experience Program
|
|
"Microsoft-Windows-DiskDiagnosticDataCollector",
|
|
|
|
# This task shows various Map related toasts
|
|
"MapsToastTask",
|
|
|
|
# This task checks for updates to maps which you have downloaded for offline use
|
|
"MapsUpdateTask",
|
|
|
|
# Initializes Family Safety monitoring and enforcement
|
|
"FamilySafetyMonitor",
|
|
|
|
# Synchronizes the latest settings with the Microsoft family features service
|
|
"FamilySafetyRefreshTask",
|
|
|
|
# XblGameSave Standby Task
|
|
"XblGameSaveTask"
|
|
)
|
|
|
|
# Check if device has a camera
|
|
$DeviceHasCamera = Get-CimInstance -ClassName Win32_PnPEntity | Where-Object -FilterScript {(($_.PNPClass -eq "Camera") -or ($_.PNPClass -eq "Image")) -and ($_.Service -ne "StillCam")}
|
|
if (-not $DeviceHasCamera)
|
|
{
|
|
# Windows Hello
|
|
$CheckedScheduledTasks += "FODCleanupTask"
|
|
}
|
|
#endregion Variables
|
|
|
|
#region XAML Markup
|
|
# The section defines the design of the upcoming dialog box
|
|
[xml]$XAML = '
|
|
<Window
|
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
|
Name="Window"
|
|
MinHeight="450" MinWidth="400"
|
|
SizeToContent="WidthAndHeight" WindowStartupLocation="CenterScreen"
|
|
TextOptions.TextFormattingMode="Display" SnapsToDevicePixels="True"
|
|
FontFamily="Candara" FontSize="16" ShowInTaskbar="True"
|
|
Background="#F1F1F1" Foreground="#262626">
|
|
<Window.Resources>
|
|
<Style TargetType="StackPanel">
|
|
<Setter Property="Orientation" Value="Horizontal"/>
|
|
<Setter Property="VerticalAlignment" Value="Top"/>
|
|
</Style>
|
|
<Style TargetType="CheckBox">
|
|
<Setter Property="Margin" Value="10, 10, 5, 10"/>
|
|
<Setter Property="IsChecked" Value="True"/>
|
|
</Style>
|
|
<Style TargetType="TextBlock">
|
|
<Setter Property="Margin" Value="5, 10, 10, 10"/>
|
|
</Style>
|
|
<Style TargetType="Button">
|
|
<Setter Property="Margin" Value="20"/>
|
|
<Setter Property="Padding" Value="10"/>
|
|
</Style>
|
|
<Style TargetType="Border">
|
|
<Setter Property="Grid.Row" Value="1"/>
|
|
<Setter Property="CornerRadius" Value="0"/>
|
|
<Setter Property="BorderThickness" Value="0, 1, 0, 1"/>
|
|
<Setter Property="BorderBrush" Value="#000000"/>
|
|
</Style>
|
|
<Style TargetType="ScrollViewer">
|
|
<Setter Property="HorizontalScrollBarVisibility" Value="Disabled"/>
|
|
<Setter Property="BorderBrush" Value="#000000"/>
|
|
<Setter Property="BorderThickness" Value="0, 1, 0, 1"/>
|
|
</Style>
|
|
</Window.Resources>
|
|
<Grid>
|
|
<Grid.RowDefinitions>
|
|
<RowDefinition Height="Auto"/>
|
|
<RowDefinition Height="*"/>
|
|
<RowDefinition Height="Auto"/>
|
|
</Grid.RowDefinitions>
|
|
<ScrollViewer Name="Scroll" Grid.Row="0"
|
|
HorizontalScrollBarVisibility="Disabled"
|
|
VerticalScrollBarVisibility="Auto">
|
|
<StackPanel Name="PanelContainer" Orientation="Vertical"/>
|
|
</ScrollViewer>
|
|
<Button Name="Button" Grid.Row="2"/>
|
|
</Grid>
|
|
</Window>
|
|
'
|
|
#endregion XAML Markup
|
|
|
|
$Reader = (New-Object -TypeName System.Xml.XmlNodeReader -ArgumentList $XAML)
|
|
$Form = [Windows.Markup.XamlReader]::Load($Reader)
|
|
$XAML.SelectNodes("//*[@*[contains(translate(name(.),'n','N'),'Name')]]") | ForEach-Object -Process {
|
|
Set-Variable -Name ($_.Name) -Value $Form.FindName($_.Name)
|
|
}
|
|
|
|
#region Functions
|
|
function Get-CheckboxClicked
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ValueFromPipeline = $true
|
|
)]
|
|
[ValidateNotNull()]
|
|
$CheckBox
|
|
)
|
|
|
|
$Task = $Tasks | Where-Object -FilterScript {$_.TaskName -eq $CheckBox.Parent.Children[1].Text}
|
|
|
|
if ($CheckBox.IsChecked)
|
|
{
|
|
[void]$SelectedTasks.Add($Task)
|
|
}
|
|
else
|
|
{
|
|
[void]$SelectedTasks.Remove($Task)
|
|
}
|
|
|
|
if ($SelectedTasks.Count -gt 0)
|
|
{
|
|
$Button.IsEnabled = $true
|
|
}
|
|
else
|
|
{
|
|
$Button.IsEnabled = $false
|
|
}
|
|
}
|
|
|
|
function DisableButton
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
[void]$Window.Close()
|
|
|
|
$SelectedTasks | ForEach-Object -Process {Write-Verbose $_.TaskName -Verbose}
|
|
$SelectedTasks | Disable-ScheduledTask
|
|
}
|
|
|
|
function EnableButton
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
[void]$Window.Close()
|
|
|
|
$SelectedTasks | ForEach-Object -Process {Write-Verbose $_.TaskName -Verbose}
|
|
$SelectedTasks | Enable-ScheduledTask
|
|
}
|
|
|
|
function Add-TaskControl
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ValueFromPipeline = $true
|
|
)]
|
|
[ValidateNotNull()]
|
|
$Task
|
|
)
|
|
|
|
process
|
|
{
|
|
$CheckBox = New-Object -TypeName System.Windows.Controls.CheckBox
|
|
$CheckBox.Add_Click({Get-CheckboxClicked -CheckBox $_.Source})
|
|
|
|
$TextBlock = New-Object -TypeName System.Windows.Controls.TextBlock
|
|
$TextBlock.Text = $Task.TaskName
|
|
|
|
$StackPanel = New-Object -TypeName System.Windows.Controls.StackPanel
|
|
[void]$StackPanel.Children.Add($CheckBox)
|
|
[void]$StackPanel.Children.Add($TextBlock)
|
|
[void]$PanelContainer.Children.Add($StackPanel)
|
|
|
|
# If task checked add to the array list
|
|
if ($CheckedScheduledTasks | Where-Object -FilterScript {$Task.TaskName -match $_})
|
|
{
|
|
[void]$SelectedTasks.Add($Task)
|
|
}
|
|
else
|
|
{
|
|
$CheckBox.IsChecked = $false
|
|
}
|
|
}
|
|
}
|
|
#endregion Functions
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
|
|
$State = "Disabled"
|
|
$ButtonContent = $Localization.Enable
|
|
$ButtonAdd_Click = {EnableButton}
|
|
}
|
|
"Disable"
|
|
{
|
|
$State = "Ready"
|
|
$ButtonContent = $Localization.Disable
|
|
$ButtonAdd_Click = {DisableButton}
|
|
}
|
|
}
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
# Getting list of all scheduled tasks according to the conditions
|
|
$Tasks = Get-ScheduledTask | Where-Object -FilterScript {($_.State -eq $State) -and ($_.TaskName -in $CheckedScheduledTasks)}
|
|
|
|
if (-not ($Tasks))
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.NoData -Verbose
|
|
|
|
return
|
|
}
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.DialogBoxOpening -Verbose
|
|
|
|
#region Sendkey function
|
|
# Emulate the Backspace key sending to prevent the console window to freeze
|
|
Start-Sleep -Milliseconds 500
|
|
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
|
|
$SetForegroundWindow = @{
|
|
Namespace = "WinAPI"
|
|
Name = "ForegroundWindow"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
[DllImport("user32.dll")]
|
|
public static extern bool ShowWindowAsync(IntPtr hWnd, int nCmdShow);
|
|
|
|
[DllImport("user32.dll")]
|
|
[return: MarshalAs(UnmanagedType.Bool)]
|
|
public static extern bool SetForegroundWindow(IntPtr hWnd);
|
|
"@
|
|
}
|
|
|
|
if (-not ("WinAPI.ForegroundWindow" -as [type]))
|
|
{
|
|
Add-Type @SetForegroundWindow
|
|
}
|
|
|
|
Get-Process | Where-Object -FilterScript {(($_.ProcessName -eq "powershell") -or ($_.ProcessName -eq "WindowsTerminal")) -and ($_.MainWindowTitle -match "Sophia Script for Windows 11")} | ForEach-Object -Process {
|
|
# Show window, if minimized
|
|
[WinAPI.ForegroundWindow]::ShowWindowAsync($_.MainWindowHandle, 10)
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Force move the console window to the foreground
|
|
[WinAPI.ForegroundWindow]::SetForegroundWindow($_.MainWindowHandle)
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Emulate the Backspace key sending
|
|
[System.Windows.Forms.SendKeys]::SendWait("{BACKSPACE 1}")
|
|
}
|
|
#endregion Sendkey function
|
|
|
|
$Window.Add_Loaded({$Tasks | Add-TaskControl})
|
|
$Button.Content = $ButtonContent
|
|
$Button.Add_Click({& $ButtonAdd_Click})
|
|
|
|
$Window.Title = $Localization.ScheduledTasks
|
|
|
|
# Force move the WPF form to the foreground
|
|
$Window.Add_Loaded({$Window.Activate()})
|
|
$Form.ShowDialog() | Out-Null
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The sign-in info to automatically finish setting up device after an update
|
|
|
|
.PARAMETER Disable
|
|
Do not use sign-in info to automatically finish setting up device after an update
|
|
|
|
.PARAMETER Enable
|
|
Use sign-in info to automatically finish setting up device after an update
|
|
|
|
.EXAMPLE
|
|
SigninInfo -Disable
|
|
|
|
.EXAMPLE
|
|
SigninInfo -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function SigninInfo
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
$SID = (Get-CimInstance -ClassName Win32_UserAccount | Where-Object -FilterScript {$_.Name -eq $env:USERNAME}).SID
|
|
if (-not (Test-Path -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon\UserARSO\$SID"))
|
|
{
|
|
New-Item -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon\UserARSO\$SID" -Force
|
|
}
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon\UserARSO\$SID" -Name OptOut -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
$SID = (Get-CimInstance -ClassName Win32_UserAccount | Where-Object -FilterScript {$_.Name -eq $env:USERNAME}).SID
|
|
Remove-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon\UserARSO\$SID" -Name OptOut -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The provision to websites a locally relevant content by accessing my language list
|
|
|
|
.PARAMETER Disable
|
|
Do not let websites show me locally relevant content by accessing my language list
|
|
|
|
.PARAMETER Enable
|
|
Let websites show me locally relevant content by accessing language my list
|
|
|
|
.EXAMPLE
|
|
LanguageListAccess -Disable
|
|
|
|
.EXAMPLE
|
|
LanguageListAccess -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function LanguageListAccess
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path "HKCU:\Control Panel\International\User Profile" -Name HttpAcceptLanguageOptOut -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
Remove-ItemProperty -Path "HKCU:\Control Panel\International\User Profile" -Name HttpAcceptLanguageOptOut -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The permission for apps to show me personalized ads by using my advertising ID
|
|
|
|
.PARAMETER Disable
|
|
Do not let apps show me personalized ads by using my advertising ID
|
|
|
|
.PARAMETER Enable
|
|
Let apps show me personalized ads by using my advertising ID
|
|
|
|
.EXAMPLE
|
|
AdvertisingID -Disable
|
|
|
|
.EXAMPLE
|
|
AdvertisingID -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function AdvertisingID
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\AdvertisingInfo))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\AdvertisingInfo -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\AdvertisingInfo -Name Enabled -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\AdvertisingInfo))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\AdvertisingInfo -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\AdvertisingInfo -Name Enabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The Windows welcome experiences after updates and occasionally when I sign in to highlight what's new and suggested
|
|
|
|
.PARAMETER Hide
|
|
Hide the Windows welcome experiences after updates and occasionally when I sign in to highlight what's new and suggested
|
|
|
|
.PARAMETER Show
|
|
Show the Windows welcome experiences after updates and occasionally when I sign in to highlight what's new and suggested
|
|
|
|
.EXAMPLE
|
|
WindowsWelcomeExperience -Hide
|
|
|
|
.EXAMPLE
|
|
WindowsWelcomeExperience -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function WindowsWelcomeExperience
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SubscribedContent-310093Enabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SubscribedContent-310093Enabled -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Getting tip and suggestions when I use Windows
|
|
|
|
.PARAMETER Enable
|
|
Get tip and suggestions when I use Windows
|
|
|
|
.PARAMETER Disable
|
|
Do not get tip and suggestions when I use Windows
|
|
|
|
.EXAMPLE
|
|
WindowsTips -Enable
|
|
|
|
.EXAMPLE
|
|
WindowsTips -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function WindowsTips
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SubscribedContent-338389Enabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SubscribedContent-338389Enabled -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Suggested me content in the Settings app
|
|
|
|
.PARAMETER Hide
|
|
Hide from me suggested content in the Settings app
|
|
|
|
.PARAMETER Show
|
|
Show me suggested content in the Settings app
|
|
|
|
.EXAMPLE
|
|
SettingsSuggestedContent -Hide
|
|
|
|
.EXAMPLE
|
|
SettingsSuggestedContent -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function SettingsSuggestedContent
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SubscribedContent-338393Enabled -PropertyType DWord -Value 0 -Force
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SubscribedContent-353694Enabled -PropertyType DWord -Value 0 -Force
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SubscribedContent-353696Enabled -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SubscribedContent-338393Enabled -PropertyType DWord -Value 1 -Force
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SubscribedContent-353694Enabled -PropertyType DWord -Value 1 -Force
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SubscribedContent-353696Enabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Automatic installing suggested apps
|
|
|
|
.PARAMETER Disable
|
|
Turn off automatic installing suggested apps
|
|
|
|
.PARAMETER Enable
|
|
Turn on automatic installing suggested apps
|
|
|
|
.EXAMPLE
|
|
AppsSilentInstalling -Disable
|
|
|
|
.EXAMPLE
|
|
AppsSilentInstalling -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function AppsSilentInstalling
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SilentInstalledAppsEnabled -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SilentInstalledAppsEnabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Suggestions on how I can set up my device
|
|
|
|
.PARAMETER Disable
|
|
Disable suggestions on how I can set up my device
|
|
|
|
.PARAMETER Enable
|
|
Offer suggestions on how I can set up my device
|
|
|
|
.EXAMPLE
|
|
WhatsNewInWindows -Disable
|
|
|
|
.EXAMPLE
|
|
WhatsNewInWindows -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function WhatsNewInWindows
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\UserProfileEngagement))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\UserProfileEngagement -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\UserProfileEngagement -Name ScoobeSystemSettingEnabled -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\UserProfileEngagement))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\UserProfileEngagement -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\UserProfileEngagement -Name ScoobeSystemSettingEnabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Tailored experiences
|
|
|
|
.PARAMETER Disable
|
|
Do not let Microsoft use your diagnostic data for personalized tips, ads, and recommendations
|
|
|
|
.PARAMETER Enable
|
|
Let Microsoft use your diagnostic data for personalized tips, ads, and recommendations
|
|
|
|
.EXAMPLE
|
|
TailoredExperiences -Disable
|
|
|
|
.EXAMPLE
|
|
TailoredExperiences -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function TailoredExperiences
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Privacy -Name TailoredExperiencesWithDiagnosticDataEnabled -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Privacy -Name TailoredExperiencesWithDiagnosticDataEnabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Bing search in the Start Menu
|
|
|
|
.PARAMETER Disable
|
|
Disable Bing search in the Start Menu
|
|
|
|
.PARAMETER Enable
|
|
Enable Bing search in the Start Menu
|
|
|
|
.EXAMPLE
|
|
BingSearch -Disable
|
|
|
|
.EXAMPLE
|
|
BingSearch -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function BingSearch
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Policies\Microsoft\Windows\Explorer))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name DisableSearchBoxSuggestions -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
Remove-ItemProperty -Path HKCU:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name DisableSearchBoxSuggestions -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
#endregion Privacy & Telemetry
|
|
|
|
#region UI & Personalization
|
|
<#
|
|
.SYNOPSIS
|
|
The "This PC" icon on Desktop
|
|
|
|
.PARAMETER Show
|
|
Show the "This PC" icon on Desktop
|
|
|
|
.PARAMETER Hide
|
|
Hide the "This PC" icon on Desktop
|
|
|
|
.EXAMPLE
|
|
ThisPC -Show
|
|
|
|
.EXAMPLE
|
|
ThisPC -Hide
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function ThisPC
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Show"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\HideDesktopIcons\NewStartPanel))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\HideDesktopIcons\NewStartPanel -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\HideDesktopIcons\NewStartPanel -Name "{20D04FE0-3AEA-1069-A2D8-08002B30309D}" -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Hide"
|
|
{
|
|
Remove-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\HideDesktopIcons\NewStartPanel -Name "{20D04FE0-3AEA-1069-A2D8-08002B30309D}" -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Item check boxes
|
|
|
|
.PARAMETER Disable
|
|
Do not use item check boxes
|
|
|
|
.PARAMETER Enable
|
|
Use check item check boxes
|
|
|
|
.EXAMPLE
|
|
CheckBoxes -Disable
|
|
|
|
.EXAMPLE
|
|
CheckBoxes -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function CheckBoxes
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name AutoCheckSelect -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name AutoCheckSelect -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Hidden files, folders, and drives
|
|
|
|
.PARAMETER Enable
|
|
Show hidden files, folders, and drives
|
|
|
|
.PARAMETER Disable
|
|
Do not show hidden files, folders, and drives
|
|
|
|
.EXAMPLE
|
|
HiddenItems -Enable
|
|
|
|
.EXAMPLE
|
|
HiddenItems -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function HiddenItems
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name Hidden -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name Hidden -PropertyType DWord -Value 2 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
File name extensions
|
|
|
|
.PARAMETER Show
|
|
Show the file name extensions
|
|
|
|
.PARAMETER Hide
|
|
Hide the file name extensions
|
|
|
|
.EXAMPLE
|
|
FileExtensions -Show
|
|
|
|
.EXAMPLE
|
|
FileExtensions -Hide
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function FileExtensions
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name HideFileExt -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name HideFileExt -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Folder merge conflicts
|
|
|
|
.PARAMETER Show
|
|
Show folder merge conflicts
|
|
|
|
.PARAMETER Hide
|
|
Hide folder merge conflicts
|
|
|
|
.EXAMPLE
|
|
MergeConflicts -Show
|
|
|
|
.EXAMPLE
|
|
MergeConflicts -Hide
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function MergeConflicts
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name HideMergeConflicts -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name HideMergeConflicts -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Configure how to open File Explorer
|
|
|
|
.PARAMETER ThisPC
|
|
Open File Explorer to "This PC"
|
|
|
|
.PARAMETER QuickAccess
|
|
Open File Explorer to Quick access
|
|
|
|
.EXAMPLE
|
|
OpenFileExplorerTo -ThisPC
|
|
|
|
.EXAMPLE
|
|
OpenFileExplorerTo -QuickAccess
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function OpenFileExplorerTo
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "ThisPC"
|
|
)]
|
|
[switch]
|
|
$ThisPC,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "QuickAccess"
|
|
)]
|
|
[switch]
|
|
$QuickAccess
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"ThisPC"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name LaunchTo -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"QuickAccess"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name LaunchTo -PropertyType DWord -Value 2 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
File Explorer mode
|
|
|
|
.PARAMETER Disable
|
|
Disable the File Explorer compact mode
|
|
|
|
.PARAMETER Enable
|
|
Enable the File Explorer compact mode
|
|
|
|
.EXAMPLE
|
|
FileExplorerCompactMode -Disable
|
|
|
|
.EXAMPLE
|
|
FileExplorerCompactMode -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function FileExplorerCompactMode
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name UseCompactMode -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name UseCompactMode -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Sync provider notification in File Explorer
|
|
|
|
.PARAMETER Hide
|
|
Do not show sync provider notification within File Explorer
|
|
|
|
.PARAMETER Show
|
|
Show sync provider notification within File Explorer
|
|
|
|
.EXAMPLE
|
|
OneDriveFileExplorerAd -Hide
|
|
|
|
.EXAMPLE
|
|
OneDriveFileExplorerAd -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function OneDriveFileExplorerAd
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name ShowSyncProviderNotifications -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name ShowSyncProviderNotifications -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Windows snapping
|
|
|
|
.PARAMETER Disable
|
|
When I snap a window, do not show what I can snap next to it
|
|
|
|
.PARAMETER Enable
|
|
When I snap a window, show what I can snap next to it
|
|
|
|
.EXAMPLE
|
|
SnapAssist -Disable
|
|
|
|
.EXAMPLE
|
|
SnapAssist -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function SnapAssist
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name SnapAssist -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name SnapAssist -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Snap layouts
|
|
|
|
.PARAMETER Enable
|
|
Show snap layouts when I hover over a windows's maximaze button
|
|
|
|
.PARAMETER Disable
|
|
Hide snap layouts when I hover over a windows's maximaze button
|
|
|
|
.EXAMPLE
|
|
SnapAssistFlyout -Enable
|
|
|
|
.EXAMPLE
|
|
SnapAssistFlyout -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function SnapAssistFlyout
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name EnableSnapAssistFlyout -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name EnableSnapAssistFlyout -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The file transfer dialog box mode
|
|
|
|
.PARAMETER Detailed
|
|
Show the file transfer dialog box in the detailed mode
|
|
|
|
.PARAMETER Compact
|
|
Show the file transfer dialog box in the compact mode
|
|
|
|
.EXAMPLE
|
|
FileTransferDialog -Detailed
|
|
|
|
.EXAMPLE
|
|
FileTransferDialog -Compact
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function FileTransferDialog
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Detailed"
|
|
)]
|
|
[switch]
|
|
$Detailed,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Compact"
|
|
)]
|
|
[switch]
|
|
$Compact
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Detailed"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\OperationStatusManager))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\OperationStatusManager -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\OperationStatusManager -Name EnthusiastMode -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Compact"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\OperationStatusManager))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\OperationStatusManager -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\OperationStatusManager -Name EnthusiastMode -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The recycle bin files delete confirmation dialog
|
|
|
|
.PARAMETER Enable
|
|
Display the recycle bin files delete confirmation dialog
|
|
|
|
.PARAMETER Disable
|
|
Do not display the recycle bin files delete confirmation dialog
|
|
|
|
.EXAMPLE
|
|
RecycleBinDeleteConfirmation -Enable
|
|
|
|
.EXAMPLE
|
|
RecycleBinDeleteConfirmation -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function RecycleBinDeleteConfirmation
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
$ShellState = Get-ItemPropertyValue -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer -Name ShellState
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
$ShellState[4] = 51
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer -Name ShellState -PropertyType Binary -Value $ShellState -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
$ShellState[4] = 55
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer -Name ShellState -PropertyType Binary -Value $ShellState -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Recently used files in Quick access
|
|
|
|
.PARAMETER Hide
|
|
Hide recently used files in Quick access
|
|
|
|
.PARAMETER Show
|
|
Show recently used files in Quick access
|
|
|
|
.EXAMPLE
|
|
QuickAccessRecentFiles -Hide
|
|
|
|
.EXAMPLE
|
|
QuickAccessRecentFiles -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function QuickAccessRecentFiles
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer -Name ShowRecent -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer -Name ShowRecent -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Frequently used folders in Quick access
|
|
|
|
.PARAMETER Hide
|
|
Hide frequently used folders in Quick access
|
|
|
|
.PARAMETER Show
|
|
Show frequently used folders in Quick access
|
|
|
|
.EXAMPLE
|
|
QuickAccessFrequentFolders -Hide
|
|
|
|
.EXAMPLE
|
|
QuickAccessFrequentFolders -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function QuickAccessFrequentFolders
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer -Name ShowFrequent -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer -Name ShowFrequent -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Taskbar alignment
|
|
|
|
.PARAMETER Left
|
|
Set the taskbar alignment to the left
|
|
|
|
.PARAMETER Center
|
|
Set the taskbar alignment to the center
|
|
|
|
.EXAMPLE
|
|
TaskbarAlignment -Center
|
|
|
|
.EXAMPLE
|
|
TaskbarAlignment -Left
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function TaskbarAlignment
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Left"
|
|
)]
|
|
[switch]
|
|
$Left,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Center"
|
|
)]
|
|
[switch]
|
|
$Center
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Center"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name TaskbarAl -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Left"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name TaskbarAl -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The search icon on the taskbar
|
|
|
|
.PARAMETER Hide
|
|
Hide the search icon on the taskbar
|
|
|
|
.PARAMETER Show
|
|
Show the search icon on the taskbar
|
|
|
|
.EXAMPLE
|
|
TaskbarSearch -Hide
|
|
|
|
.EXAMPLE
|
|
TaskbarSearch -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function TaskbarSearch
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Search -Name SearchboxTaskbarMode -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Search -Name SearchboxTaskbarMode -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Task view button on the taskbar
|
|
|
|
.PARAMETER Hide
|
|
Hide the Task view button on the taskbar
|
|
|
|
.PARAMETER Show
|
|
Show the Task View button on the taskbar
|
|
|
|
.EXAMPLE
|
|
TaskViewButton -Hide
|
|
|
|
.EXAMPLE
|
|
TaskViewButton -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function TaskViewButton
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name ShowTaskViewButton -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name ShowTaskViewButton -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The widgets icon on the taskbar
|
|
|
|
.PARAMETER Hide
|
|
Hide the widgets icon on the taskbar
|
|
|
|
.PARAMETER Show
|
|
Show the widgets icon on the taskbar
|
|
|
|
.EXAMPLE
|
|
TaskbarWidgets -Hide
|
|
|
|
.EXAMPLE
|
|
TaskbarWidgets -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function TaskbarWidgets
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
if (Get-AppxPackage -Name MicrosoftWindows.Client.WebExperience)
|
|
{
|
|
New-ItemProperty -Path HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name TaskbarDa -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
"Show"
|
|
{
|
|
if (Get-AppxPackage -Name MicrosoftWindows.Client.WebExperience)
|
|
{
|
|
New-ItemProperty -Path HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name TaskbarDa -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The Chat icon (Microsoft Teams) on the taskbar
|
|
|
|
.PARAMETER Hide
|
|
Hide the Chat icon (Microsoft Teams) on the taskbar
|
|
|
|
.PARAMETER Show
|
|
Show the Chat icon (Microsoft Teams) on the taskbar
|
|
|
|
.EXAMPLE
|
|
TaskbarChat -Hide
|
|
|
|
.EXAMPLE
|
|
TaskbarChat -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function TaskbarChat
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name TaskbarMn -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name TaskbarMn -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Unpin shortcuts from the taskbar
|
|
|
|
.PARAMETER Edge
|
|
Unpin the "Microsoft Edge" shortcut from the taskbar
|
|
|
|
.PARAMETER Store
|
|
Unpin the "Microsoft Store" shortcut from the taskbar
|
|
|
|
.EXAMPLE
|
|
UnpinTaskbarShortcuts -Shortcuts Edge, Store
|
|
|
|
.NOTES
|
|
Current user
|
|
|
|
.LINK
|
|
https://github.com/Disassembler0/Win10-Initial-Setup-Script/issues/8#issue-227159084
|
|
#>
|
|
function UnpinTaskbarShortcuts
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(Mandatory = $true)]
|
|
[ValidateSet("Edge", "Store")]
|
|
[string[]]
|
|
$Shortcuts
|
|
)
|
|
|
|
# Extract strings from shell32.dll using its' number
|
|
$Signature = @{
|
|
Namespace = "WinAPI"
|
|
Name = "GetStr"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
[DllImport("kernel32.dll", CharSet = CharSet.Auto)]
|
|
public static extern IntPtr GetModuleHandle(string lpModuleName);
|
|
|
|
[DllImport("user32.dll", CharSet = CharSet.Auto)]
|
|
internal static extern int LoadString(IntPtr hInstance, uint uID, StringBuilder lpBuffer, int nBufferMax);
|
|
|
|
public static string GetString(uint strId)
|
|
{
|
|
IntPtr intPtr = GetModuleHandle("shell32.dll");
|
|
StringBuilder sb = new StringBuilder(255);
|
|
LoadString(intPtr, strId, sb, sb.Capacity);
|
|
return sb.ToString();
|
|
}
|
|
"@
|
|
}
|
|
if (-not ("WinAPI.GetStr" -as [type]))
|
|
{
|
|
Add-Type @Signature -Using System.Text
|
|
}
|
|
|
|
# Extract the localized "Unpin from taskbar" string from shell32.dll
|
|
$LocalizedString = [WinAPI.GetStr]::GetString(5387)
|
|
|
|
foreach ($Shortcut in $Shortcuts)
|
|
{
|
|
switch ($Shortcut)
|
|
{
|
|
Edge
|
|
{
|
|
if (Test-Path -Path "$env:AppData\Microsoft\Internet Explorer\Quick Launch\User Pinned\TaskBar\Microsoft Edge.lnk")
|
|
{
|
|
# Call the shortcut context menu item
|
|
$Shell = (New-Object -ComObject Shell.Application).NameSpace("$env:AppData\Microsoft\Internet Explorer\Quick Launch\User Pinned\TaskBar")
|
|
$Shortcut = $Shell.ParseName("Microsoft Edge.lnk")
|
|
$Shortcut.Verbs() | Where-Object -FilterScript {$_.Name -eq $LocalizedString} | ForEach-Object -Process {$_.DoIt()}
|
|
}
|
|
}
|
|
Store
|
|
{
|
|
# Start-Job is used due to that the calling this function before UninstallUWPApps breaks the retrieval of the localized UWP apps packages names
|
|
Start-Job -ScriptBlock {
|
|
$Apps = (New-Object -ComObject Shell.Application).NameSpace("shell:::{4234d49b-0245-4df3-b780-3893943456e1}").Items()
|
|
($Apps | Where-Object -FilterScript {$_.Name -eq "Microsoft Store"}).Verbs() | Where-Object -FilterScript {$_.Name -eq $Using:LocalizedString} | ForEach-Object -Process {$_.DoIt()}
|
|
} | Receive-Job -Wait -AutoRemoveJob
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The Control Panel icons view
|
|
|
|
.PARAMETER Category
|
|
View the Control Panel icons by category
|
|
|
|
.PARAMETER LargeIcons
|
|
View the Control Panel icons by large icons
|
|
|
|
.PARAMETER SmallIcons
|
|
View the Control Panel icons by Small icons
|
|
|
|
.EXAMPLE
|
|
ControlPanelView -Category
|
|
|
|
.EXAMPLE
|
|
ControlPanelView -LargeIcons
|
|
|
|
.EXAMPLE
|
|
ControlPanelView -SmallIcons
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function ControlPanelView
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Category"
|
|
)]
|
|
[switch]
|
|
$Category,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "LargeIcons"
|
|
)]
|
|
[switch]
|
|
$LargeIcons,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "SmallIcons"
|
|
)]
|
|
[switch]
|
|
$SmallIcons
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Category"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\ControlPanel))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\ControlPanel -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\ControlPanel -Name AllItemsIconView -PropertyType DWord -Value 0 -Force
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\ControlPanel -Name StartupPage -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"LargeIcons"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\ControlPanel))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\ControlPanel -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\ControlPanel -Name AllItemsIconView -PropertyType DWord -Value 0 -Force
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\ControlPanel -Name StartupPage -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"SmallIcons"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\ControlPanel))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\ControlPanel -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\ControlPanel -Name AllItemsIconView -PropertyType DWord -Value 1 -Force
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\ControlPanel -Name StartupPage -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The default Windows mode
|
|
|
|
.PARAMETER Dark
|
|
Set the default Windows mode to dark
|
|
|
|
.PARAMETER Light
|
|
Set the default Windows mode to light
|
|
|
|
.EXAMPLE
|
|
WindowsColorScheme -Dark
|
|
|
|
.EXAMPLE
|
|
WindowsColorScheme -Light
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function WindowsColorMode
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Dark"
|
|
)]
|
|
[switch]
|
|
$Dark,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Light"
|
|
)]
|
|
[switch]
|
|
$Light
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Dark"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Themes\Personalize -Name SystemUsesLightTheme -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Light"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Themes\Personalize -Name SystemUsesLightTheme -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The default app mode
|
|
|
|
.PARAMETER Dark
|
|
Set the default app mode to dark
|
|
|
|
.PARAMETER Light
|
|
Set the default app mode to light
|
|
|
|
.EXAMPLE
|
|
AppColorMode -Dark
|
|
|
|
.EXAMPLE
|
|
AppColorMode -Light
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function AppColorMode
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Dark"
|
|
)]
|
|
[switch]
|
|
$Dark,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Light"
|
|
)]
|
|
[switch]
|
|
$Light
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Dark"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Themes\Personalize -Name AppsUseLightTheme -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Light"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Themes\Personalize -Name AppsUseLightTheme -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
First sign-in animation after the upgrade
|
|
|
|
.PARAMETER Disable
|
|
Disable first sign-in animation after the upgrade
|
|
|
|
.PARAMETER Enable
|
|
Enable first sign-in animation after the upgrade
|
|
|
|
.EXAMPLE
|
|
FirstLogonAnimation -Disable
|
|
|
|
.EXAMPLE
|
|
FirstLogonAnimation -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function FirstLogonAnimation
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon" -Name EnableFirstLogonAnimation -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon" -Name EnableFirstLogonAnimation -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The quality factor of the JPEG desktop wallpapers
|
|
|
|
.PARAMETER Max
|
|
Set the quality factor of the JPEG desktop wallpapers to maximum
|
|
|
|
.PARAMETER Default
|
|
Set the quality factor of the JPEG desktop wallpapers to default
|
|
|
|
.EXAMPLE
|
|
JPEGWallpapersQuality -Max
|
|
|
|
.EXAMPLE
|
|
JPEGWallpapersQuality -Default
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function JPEGWallpapersQuality
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Max"
|
|
)]
|
|
[switch]
|
|
$Max,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Default"
|
|
)]
|
|
[switch]
|
|
$Default
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Max"
|
|
{
|
|
New-ItemProperty -Path "HKCU:\Control Panel\Desktop" -Name JPEGImportQuality -PropertyType DWord -Value 100 -Force
|
|
}
|
|
"Default"
|
|
{
|
|
Remove-ItemProperty -Path "HKCU:\Control Panel\Desktop" -Name JPEGImportQuality -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The Task Manager mode
|
|
|
|
.PARAMETER Expanded
|
|
Start Task Manager in the expanded mode
|
|
|
|
.PARAMETER Compact
|
|
Start Task Manager in the compact mode
|
|
|
|
.EXAMPLE
|
|
TaskManagerWindow -Expanded
|
|
|
|
.EXAMPLE
|
|
TaskManagerWindow -Compact
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function TaskManagerWindow
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Expanded"
|
|
)]
|
|
[switch]
|
|
$Expanded,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Compact"
|
|
)]
|
|
[switch]
|
|
$Compact
|
|
)
|
|
|
|
# This function is designed only for 22000 build due to Windows 11 22H2 has a new Task Manager with the new UI
|
|
if ((Get-CimInstance -ClassName Win32_OperatingSystem).BuildNumber -eq 22000)
|
|
{
|
|
$Taskmgr = Get-Process -Name Taskmgr -ErrorAction Ignore
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
if ($Taskmgr)
|
|
{
|
|
$Taskmgr.CloseMainWindow()
|
|
}
|
|
Start-Process -FilePath Taskmgr.exe -PassThru
|
|
|
|
Start-Sleep -Seconds 3
|
|
|
|
do
|
|
{
|
|
Start-Sleep -Milliseconds 100
|
|
$Preferences = Get-ItemPropertyValue -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\TaskManager -Name Preferences
|
|
}
|
|
until ($Preferences)
|
|
|
|
Stop-Process -Name Taskmgr -ErrorAction Ignore
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Expanded"
|
|
{
|
|
$Preferences[28] = 0
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\TaskManager -Name Preferences -PropertyType Binary -Value $Preferences -Force
|
|
}
|
|
"Compact"
|
|
{
|
|
$Preferences[28] = 1
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\TaskManager -Name Preferences -PropertyType Binary -Value $Preferences -Force
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Notification when your PC requires a restart to finish updating
|
|
|
|
.PARAMETER Show
|
|
Notify me when a restart is required to finish updatingg
|
|
|
|
.PARAMETER Hide
|
|
Do not notify me when a restart is required to finish updating
|
|
|
|
.EXAMPLE
|
|
RestartNotification -Show
|
|
|
|
.EXAMPLE
|
|
RestartNotification -Hide
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function RestartNotification
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\WindowsUpdate\UX\Settings -Name RestartNotificationsAllowed2 -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\WindowsUpdate\UX\Settings -Name RestartNotificationsAllowed2 -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "- Shortcut" suffix adding to the name of the created shortcuts
|
|
|
|
.PARAMETER Disable
|
|
Do not add the "- Shortcut" suffix to the file name of created shortcuts
|
|
|
|
.PARAMETER Enable
|
|
Add the "- Shortcut" suffix to the file name of created shortcuts
|
|
|
|
.EXAMPLE
|
|
ShortcutsSuffix -Disable
|
|
|
|
.EXAMPLE
|
|
ShortcutsSuffix -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function ShortcutsSuffix
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\NamingTemplates))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\NamingTemplates -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\NamingTemplates -Name ShortcutNameTemplate -PropertyType String -Value "%s.lnk" -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
Remove-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\NamingTemplates -Name ShortcutNameTemplate -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The Print screen button usage
|
|
|
|
.PARAMETER Enable
|
|
Use the Print screen button to open screen snipping
|
|
|
|
.PARAMETER Disable
|
|
Do not use the Print screen button to open screen snipping
|
|
|
|
.EXAMPLE
|
|
PrtScnSnippingTool -Enable
|
|
|
|
.EXAMPLE
|
|
PrtScnSnippingTool -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function PrtScnSnippingTool
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path "HKCU:\Control Panel\Keyboard" -Name PrintScreenKeyForSnippingEnabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path "HKCU:\Control Panel\Keyboard" -Name PrintScreenKeyForSnippingEnabled -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
A different input method for each app window
|
|
|
|
.PARAMETER Enable
|
|
Let me use a different input method for each app window
|
|
|
|
.PARAMETER Disable
|
|
Do not use a different input method for each app window
|
|
|
|
.EXAMPLE
|
|
AppsLanguageSwitch -Enable
|
|
|
|
.EXAMPLE
|
|
AppsLanguageSwitch -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function AppsLanguageSwitch
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
Set-WinLanguageBarOption -UseLegacySwitchMode
|
|
}
|
|
"Disable"
|
|
{
|
|
Set-WinLanguageBarOption
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Title bar window shake
|
|
|
|
.PARAMETER Enable
|
|
When I grab a windows's title bar and shake it, minimize all other windows
|
|
|
|
.PARAMETER Disable
|
|
When I grab a windows's title bar and shake it, don't minimize all other windows
|
|
|
|
.EXAMPLE
|
|
AeroShaking -Enable
|
|
|
|
.EXAMPLE
|
|
AeroShaking -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function AeroShaking
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name DisallowShaking -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name DisallowShaking -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
#endregion UI & Personalization
|
|
|
|
#region OneDrive
|
|
<#
|
|
.SYNOPSIS
|
|
OneDrive
|
|
|
|
.PARAMETER Uninstall
|
|
Uninstall OneDrive
|
|
|
|
.PARAMETER Install
|
|
Install OneDrive 64-bit
|
|
|
|
.EXAMPLE
|
|
OneDrive -Uninstall
|
|
|
|
.EXAMPLE
|
|
OneDrive -Install
|
|
|
|
.NOTES
|
|
The OneDrive user folder won't be removed
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function OneDrive
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Uninstall"
|
|
)]
|
|
[switch]
|
|
$Uninstall,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Install"
|
|
)]
|
|
[switch]
|
|
$Install
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Uninstall"
|
|
{
|
|
[string]$UninstallString = Get-Package -Name "Microsoft OneDrive" -ProviderName Programs -ErrorAction Ignore | ForEach-Object -Process {$_.Meta.Attributes["UninstallString"]}
|
|
if ($UninstallString)
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.OneDriveUninstalling -Verbose
|
|
|
|
Stop-Process -Name OneDrive -Force -ErrorAction Ignore
|
|
Stop-Process -Name OneDriveSetup -Force -ErrorAction Ignore
|
|
Stop-Process -Name FileCoAuth -Force -ErrorAction Ignore
|
|
|
|
# Getting link to the OneDriveSetup.exe and its' argument(s)
|
|
[string[]]$OneDriveSetup = ($UninstallString -Replace("\s*/", ",/")).Split(",").Trim()
|
|
if ($OneDriveSetup.Count -eq 2)
|
|
{
|
|
Start-Process -FilePath $OneDriveSetup[0] -ArgumentList $OneDriveSetup[1..1] -Wait
|
|
}
|
|
else
|
|
{
|
|
Start-Process -FilePath $OneDriveSetup[0] -ArgumentList $OneDriveSetup[1..2] -Wait
|
|
}
|
|
|
|
# Get the OneDrive user folder path and remove it if it doesn't contain any user files
|
|
if (Test-Path -Path $env:OneDrive)
|
|
{
|
|
if ((Get-ChildItem -Path $env:OneDrive -ErrorAction Ignore | Measure-Object).Count -eq 0)
|
|
{
|
|
Remove-Item -Path $env:OneDrive -Recurse -Force -ErrorAction Ignore
|
|
|
|
# https://docs.microsoft.com/en-us/windows/win32/api/winbase/nf-winbase-movefileexa
|
|
# The system does not move the file until the operating system is restarted
|
|
# The system moves the file immediately after AUTOCHK is executed, but before creating any paging files
|
|
$Signature = @{
|
|
Namespace = "WinAPI"
|
|
Name = "DeleteFiles"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
public enum MoveFileFlags
|
|
{
|
|
MOVEFILE_DELAY_UNTIL_REBOOT = 0x00000004
|
|
}
|
|
|
|
[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
|
|
static extern bool MoveFileEx(string lpExistingFileName, string lpNewFileName, MoveFileFlags dwFlags);
|
|
|
|
public static bool MarkFileDelete (string sourcefile)
|
|
{
|
|
return MoveFileEx(sourcefile, null, MoveFileFlags.MOVEFILE_DELAY_UNTIL_REBOOT);
|
|
}
|
|
"@
|
|
}
|
|
|
|
# If there are some files or folders left in %OneDrive%
|
|
if ((Get-ChildItem -Path $env:OneDrive -ErrorAction Ignore | Measure-Object).Count -ne 0)
|
|
{
|
|
if (-not ("WinAPI.DeleteFiles" -as [type]))
|
|
{
|
|
Add-Type @Signature
|
|
}
|
|
|
|
try
|
|
{
|
|
Remove-Item -Path $env:OneDrive -Recurse -Force -ErrorAction Stop
|
|
}
|
|
catch
|
|
{
|
|
# If files are in use remove them at the next boot
|
|
Get-ChildItem -Path $env:OneDrive -Recurse -Force | ForEach-Object -Process {[WinAPI.DeleteFiles]::MarkFileDelete($_.FullName)}
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
Start-Process -FilePath explorer -ArgumentList $env:OneDrive
|
|
}
|
|
}
|
|
|
|
Remove-ItemProperty -Path HKCU:\Environment -Name OneDrive, OneDriveConsumer -Force -ErrorAction Ignore
|
|
Remove-Item -Path HKCU:\SOFTWARE\Microsoft\OneDrive -Recurse -Force -ErrorAction Ignore
|
|
Remove-Item -Path HKLM:\SOFTWARE\WOW6432Node\Microsoft\OneDrive -Recurse -Force -ErrorAction Ignore
|
|
Remove-Item -Path "$env:ProgramData\Microsoft OneDrive" -Recurse -Force -ErrorAction Ignore
|
|
Remove-Item -Path $env:SystemDrive\OneDriveTemp -Recurse -Force -ErrorAction Ignore
|
|
Unregister-ScheduledTask -TaskName *OneDrive* -Confirm:$false -ErrorAction Ignore
|
|
|
|
# Getting the OneDrive folder path
|
|
$OneDriveFolder = Split-Path -Path (Split-Path -Path $OneDriveSetup[0] -Parent)
|
|
|
|
# Save all opened folders in order to restore them after File Explorer restarting
|
|
Clear-Variable -Name OpenedFolders -Force -ErrorAction Ignore
|
|
$Script:OpenedFolders = {(New-Object -ComObject Shell.Application).Windows() | ForEach-Object -Process {$_.Document.Folder.Self.Path}}.Invoke()
|
|
|
|
# Terminate the File Explorer process
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon" -Name AutoRestartShell -PropertyType DWord -Value 0 -Force
|
|
Stop-Process -Name explorer -Force
|
|
Start-Sleep -Seconds 3
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon" -Name AutoRestartShell -PropertyType DWord -Value 1 -Force
|
|
|
|
# Attempt to unregister FileSyncShell64.dll and remove
|
|
$FileSyncShell64dlls = Get-ChildItem -Path "$OneDriveFolder\*\FileSyncShell64.dll" -Force
|
|
foreach ($FileSyncShell64dll in $FileSyncShell64dlls.FullName)
|
|
{
|
|
Start-Process -FilePath regsvr32.exe -ArgumentList "/u /s $FileSyncShell64dll" -Wait
|
|
Remove-Item -Path $FileSyncShell64dll -Force -ErrorAction Ignore
|
|
|
|
if (Test-Path -Path $FileSyncShell64dll)
|
|
{
|
|
if (-not ("WinAPI.DeleteFiles" -as [type]))
|
|
{
|
|
Add-Type @Signature
|
|
}
|
|
|
|
# If files are in use remove them at the next boot
|
|
Get-ChildItem -Path $FileSyncShell64dll -Recurse -Force | ForEach-Object -Process {[WinAPI.DeleteFiles]::MarkFileDelete($_.FullName)}
|
|
}
|
|
}
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Start the File Explorer process
|
|
Start-Process -FilePath explorer
|
|
|
|
# Restoring closed folders
|
|
foreach ($OpenedFolder in $OpenedFolders)
|
|
{
|
|
if (Test-Path -Path $OpenedFolder)
|
|
{
|
|
Start-Process -FilePath explorer -ArgumentList $OpenedFolder
|
|
}
|
|
}
|
|
|
|
Remove-Item -Path $OneDriveFolder -Recurse -Force -ErrorAction Ignore
|
|
Remove-Item -Path $env:LOCALAPPDATA\OneDrive -Recurse -Force -ErrorAction Ignore
|
|
Remove-Item -Path $env:LOCALAPPDATA\Microsoft\OneDrive -Recurse -Force -ErrorAction Ignore
|
|
Remove-Item -Path "$env:APPDATA\Microsoft\Windows\Start Menu\Programs\OneDrive.lnk" -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
"Install"
|
|
{
|
|
$OneDrive = Get-Package -Name "Microsoft OneDrive" -ProviderName Programs -Force -ErrorAction Ignore
|
|
if (-not $OneDrive)
|
|
{
|
|
if (Test-Path -Path $env:SystemRoot\SysWOW64\OneDriveSetup.exe)
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.OneDriveInstalling -Verbose
|
|
|
|
Start-Process -FilePath $env:SystemRoot\SysWOW64\OneDriveSetup.exe
|
|
}
|
|
else
|
|
{
|
|
try
|
|
{
|
|
# Check the internet connection
|
|
$Parameters = @{
|
|
Uri = "https://www.google.com"
|
|
Method = "Head"
|
|
DisableKeepAlive = $true
|
|
UseBasicParsing = $true
|
|
}
|
|
if (-not (Invoke-WebRequest @Parameters).StatusDescription)
|
|
{
|
|
return
|
|
}
|
|
|
|
# Downloading the latest OneDrive installer 64-bit
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.OneDriveDownloading -Verbose
|
|
|
|
# Parse XML to get the URL
|
|
# https://go.microsoft.com/fwlink/p/?LinkID=844652
|
|
$Parameters = @{
|
|
Uri = "https://g.live.com/1rewlive5skydrive/OneDriveProduction"
|
|
UseBasicParsing = $true
|
|
Verbose = $true
|
|
}
|
|
$Content = Invoke-RestMethod @Parameters
|
|
|
|
# Remove invalid chars
|
|
[xml]$OneDriveXML = $Content -replace "", ""
|
|
|
|
$OneDriveURL = ($OneDriveXML).root.update.amd64binary.url
|
|
$DownloadsFolder = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "{374DE290-123F-4565-9164-39C4925E467B}"
|
|
$Parameters = @{
|
|
Uri = $OneDriveURL
|
|
OutFile = "$DownloadsFolder\OneDriveSetup.exe"
|
|
UseBasicParsing = $true
|
|
Verbose = $true
|
|
}
|
|
Invoke-WebRequest @Parameters
|
|
|
|
Start-Process -FilePath "$DownloadsFolder\OneDriveSetup.exe" -Wait
|
|
|
|
Remove-Item -Path "$DownloadsFolder\OneDriveSetup.exe" -Force
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message $Localization.NoInternetConnection
|
|
Write-Error -Message $Localization.NoInternetConnection -ErrorAction SilentlyContinue
|
|
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
|
|
# Save screenshots by pressing Win+PrtScr in the Pictures folder
|
|
Remove-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "{B7BEDE81-DF94-4682-A7D8-57A52620B86F}" -Force -ErrorAction Ignore
|
|
|
|
Get-ScheduledTask -TaskName "Onedrive* Update*" | Enable-ScheduledTask
|
|
Get-ScheduledTask -TaskName "Onedrive* Update*" | Start-ScheduledTask
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#endregion OneDrive
|
|
|
|
#region System
|
|
#region StorageSense
|
|
<#
|
|
.SYNOPSIS
|
|
Storage Sense
|
|
|
|
.PARAMETER Enable
|
|
Turn on Storage Sense
|
|
|
|
.PARAMETER Disable
|
|
Turn off Storage Sense
|
|
|
|
.EXAMPLE
|
|
StorageSense -Enable
|
|
|
|
.EXAMPLE
|
|
StorageSense -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function StorageSense
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy -ItemType Directory -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy -Name 01 -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy -ItemType Directory -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy -Name 01 -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Clean up of temporary files
|
|
|
|
.PARAMETER Enable
|
|
Turn on automatic cleaning up temporary system and app files
|
|
|
|
.PARAMETER Disable
|
|
Turn off automatic cleaning up temporary system and app files
|
|
|
|
.EXAMPLE
|
|
StorageSenseTempFiles -Enable
|
|
|
|
.EXAMPLE
|
|
StorageSenseTempFiles -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function StorageSenseTempFiles
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
if ((Get-ItemPropertyValue -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy -Name 01) -eq "1")
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy -Name 04 -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
"Disable"
|
|
{
|
|
if ((Get-ItemPropertyValue -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy -Name 01) -eq "1")
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy -Name 04 -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Storage Sense running frequency
|
|
|
|
.PARAMETER Month
|
|
Run Storage Sense every month
|
|
|
|
.PARAMETER Default
|
|
Run Storage Sense during low free disk space
|
|
|
|
.EXAMPLE
|
|
StorageSenseFrequency -Month
|
|
|
|
.EXAMPLE
|
|
StorageSenseFrequency -Default
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function StorageSenseFrequency
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Month"
|
|
)]
|
|
[switch]
|
|
$Month,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Default"
|
|
)]
|
|
[switch]
|
|
$Default
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Month"
|
|
{
|
|
if ((Get-ItemPropertyValue -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy -Name 01) -eq "1")
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy -Name 2048 -PropertyType DWord -Value 30 -Force
|
|
}
|
|
}
|
|
"Default"
|
|
{
|
|
if ((Get-ItemPropertyValue -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy -Name 01) -eq "1")
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\StorageSense\Parameters\StoragePolicy -Name 2048 -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#endregion StorageSense
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Hibernation
|
|
|
|
.PARAMETER Disable
|
|
Disable hibernation
|
|
|
|
.PARAMETER Enable
|
|
Enable hibernation
|
|
|
|
.EXAMPLE
|
|
Hibernation -Enable
|
|
|
|
.EXAMPLE
|
|
Hibernation -Disable
|
|
|
|
.NOTES
|
|
Do not recommend turning it off on laptops
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function Hibernation
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
POWERCFG /HIBERNATE OFF
|
|
}
|
|
"Enable"
|
|
{
|
|
POWERCFG /HIBERNATE ON
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The %TEMP% environment variable path
|
|
|
|
.PARAMETER SystemDrive
|
|
Change the %TEMP% environment variable path to %SystemDrive%\Temp
|
|
|
|
.PARAMETER Default
|
|
Change the %TEMP% environment variable path to %LOCALAPPDATA%\Temp
|
|
|
|
.EXAMPLE
|
|
TempFolder -SystemDrive
|
|
|
|
.EXAMPLE
|
|
TempFolder -Default
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function TempFolder
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "SystemDrive"
|
|
)]
|
|
[switch]
|
|
$SystemDrive,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Default"
|
|
)]
|
|
[switch]
|
|
$Default
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"SystemDrive"
|
|
{
|
|
if ($env:TEMP -ne "$env:SystemDrive\Temp")
|
|
{
|
|
# Restart the Printer Spooler service (Spooler)
|
|
Restart-Service -Name Spooler -Force
|
|
|
|
# Stop OneDrive processes
|
|
Stop-Process -Name OneDrive -Force -ErrorAction Ignore
|
|
Stop-Process -Name FileCoAuth -Force -ErrorAction Ignore
|
|
|
|
if (-not (Test-Path -Path $env:SystemDrive\Temp))
|
|
{
|
|
New-Item -Path $env:SystemDrive\Temp -ItemType Directory -Force
|
|
}
|
|
|
|
# Cleaning up folders
|
|
Remove-Item -Path $env:SystemRoot\Temp -Recurse -Force -ErrorAction Ignore
|
|
Get-Item -Path $env:TEMP -Force -ErrorAction Ignore | Where-Object -FilterScript {$_.LinkType -ne "SymbolicLink"} | Remove-Item -Recurse -Force -ErrorAction Ignore
|
|
|
|
if (-not (Test-Path -Path $env:LOCALAPPDATA\Temp))
|
|
{
|
|
New-Item -Path $env:LOCALAPPDATA\Temp -ItemType Directory -Force
|
|
}
|
|
|
|
# If there are some files or folders left in %LOCALAPPDATA\Temp%
|
|
if ((Get-ChildItem -Path $env:TEMP -Force -ErrorAction Ignore | Measure-Object).Count -ne 0)
|
|
{
|
|
# https://docs.microsoft.com/en-us/windows/win32/api/winbase/nf-winbase-movefileexa
|
|
# The system does not move the file until the operating system is restarted
|
|
# The system moves the file immediately after AUTOCHK is executed, but before creating any paging files
|
|
$Signature = @{
|
|
Namespace = "WinAPI"
|
|
Name = "DeleteFiles"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
public enum MoveFileFlags
|
|
{
|
|
MOVEFILE_DELAY_UNTIL_REBOOT = 0x00000004
|
|
}
|
|
|
|
[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
|
|
static extern bool MoveFileEx(string lpExistingFileName, string lpNewFileName, MoveFileFlags dwFlags);
|
|
|
|
public static bool MarkFileDelete (string sourcefile)
|
|
{
|
|
return MoveFileEx(sourcefile, null, MoveFileFlags.MOVEFILE_DELAY_UNTIL_REBOOT);
|
|
}
|
|
"@
|
|
}
|
|
|
|
if (-not ("WinAPI.DeleteFiles" -as [type]))
|
|
{
|
|
Add-Type @Signature
|
|
}
|
|
|
|
try
|
|
{
|
|
Get-ChildItem -Path $env:TEMP -Recurse -Force | Remove-Item -Recurse -Force -ErrorAction Stop
|
|
}
|
|
catch
|
|
{
|
|
# If files are in use remove them at the next boot
|
|
Get-ChildItem -Path $env:TEMP -Recurse -Force | ForEach-Object -Process {[WinAPI.DeleteFiles]::MarkFileDelete($_.FullName)}
|
|
}
|
|
|
|
$SymbolicLinkTask = @"
|
|
Get-ChildItem -Path `$env:LOCALAPPDATA\Temp -Recurse -Force | Remove-Item -Recurse -Force
|
|
|
|
Get-Item -Path `$env:LOCALAPPDATA\Temp -Force | Where-Object -FilterScript {`$_.LinkType -ne """SymbolicLink"""} | Remove-Item -Recurse -Force
|
|
New-Item -Path `$env:LOCALAPPDATA\Temp -ItemType SymbolicLink -Value `$env:SystemDrive\Temp -Force
|
|
|
|
Unregister-ScheduledTask -TaskName SymbolicLink -Confirm:`$false
|
|
"@
|
|
|
|
# Create a temporary scheduled task to create a symbolic link to the %SystemDrive%\Temp folder
|
|
$Action = New-ScheduledTaskAction -Execute powershell.exe -Argument "-WindowStyle Hidden -Command $SymbolicLinkTask"
|
|
$Trigger = New-ScheduledTaskTrigger -AtLogon -User $env:USERNAME
|
|
$Settings = New-ScheduledTaskSettingsSet -Compatibility Win8
|
|
$Principal = New-ScheduledTaskPrincipal -UserId $env:USERNAME -RunLevel Highest
|
|
$Parameters = @{
|
|
TaskName = "SymbolicLink"
|
|
Principal = $Principal
|
|
Action = $Action
|
|
Settings = $Settings
|
|
Trigger = $Trigger
|
|
}
|
|
Register-ScheduledTask @Parameters -Force
|
|
}
|
|
else
|
|
{
|
|
# Create a symbolic link to the %SystemDrive%\Temp folder
|
|
New-Item -Path $env:LOCALAPPDATA\Temp -ItemType SymbolicLink -Value $env:SystemDrive\Temp -Force
|
|
}
|
|
|
|
#region main
|
|
# Change the %TEMP% environment variable path to %LOCALAPPDATA%\Temp
|
|
# The additional registry key creating are needed to fix the property type of the keys: SetEnvironmentVariable creates them with the "String" type instead of "ExpandString" as by default
|
|
[Environment]::SetEnvironmentVariable("TMP", "$env:SystemDrive\Temp", "User")
|
|
[Environment]::SetEnvironmentVariable("TMP", "$env:SystemDrive\Temp", "Machine")
|
|
[Environment]::SetEnvironmentVariable("TMP", "$env:SystemDrive\Temp", "Process")
|
|
New-ItemProperty -Path HKCU:\Environment -Name TMP -PropertyType ExpandString -Value $env:SystemDrive\Temp -Force
|
|
|
|
[Environment]::SetEnvironmentVariable("TEMP", "$env:SystemDrive\Temp", "User")
|
|
[Environment]::SetEnvironmentVariable("TEMP", "$env:SystemDrive\Temp", "Machine")
|
|
[Environment]::SetEnvironmentVariable("TEMP", "$env:SystemDrive\Temp", "Process")
|
|
New-ItemProperty -Path HKCU:\Environment -Name TEMP -PropertyType ExpandString -Value $env:SystemDrive\Temp -Force
|
|
|
|
New-ItemProperty -Path "HKLM:\SYSTEM\CurrentControlSet\Control\Session Manager\Environment" -Name TMP -PropertyType ExpandString -Value $env:SystemDrive\Temp -Force
|
|
New-ItemProperty -Path "HKLM:\SYSTEM\CurrentControlSet\Control\Session Manager\Environment" -Name TEMP -PropertyType ExpandString -Value $env:SystemDrive\Temp -Force
|
|
# endregion main
|
|
}
|
|
}
|
|
"Default"
|
|
{
|
|
if ($env:TEMP -ne "$env:LOCALAPPDATA\Temp")
|
|
{
|
|
# Restart the Printer Spooler service (Spooler)
|
|
Restart-Service -Name Spooler -Force
|
|
|
|
# Stop OneDrive processes
|
|
Stop-Process -Name OneDrive -Force -ErrorAction Ignore
|
|
Stop-Process -Name FileCoAuth -Force -ErrorAction Ignore
|
|
|
|
# Remove a symbolic link to the %SystemDrive%\Temp folder
|
|
if (Get-Item -Path $env:LOCALAPPDATA\Temp -Force -ErrorAction Ignore | Where-Object -FilterScript {$_.LinkType -eq "SymbolicLink"})
|
|
{
|
|
(Get-Item -Path $env:LOCALAPPDATA\Temp -Force).Delete()
|
|
}
|
|
|
|
if (-not (Test-Path -Path $env:SystemRoot\Temp))
|
|
{
|
|
New-Item -Path $env:SystemRoot\Temp -ItemType Directory -Force
|
|
}
|
|
if (-not (Test-Path -Path $env:LOCALAPPDATA\Temp))
|
|
{
|
|
New-Item -Path $env:LOCALAPPDATA\Temp -ItemType Directory -Force
|
|
}
|
|
|
|
# Removing folders
|
|
Remove-Item -Path $env:TEMP -Recurse -Force -ErrorAction Ignore
|
|
|
|
if ((Get-ChildItem -Path $env:TEMP -Force -ErrorAction Ignore | Measure-Object).Count -ne 0)
|
|
{
|
|
# https://docs.microsoft.com/en-us/windows/win32/api/winbase/nf-winbase-movefileexa
|
|
# The system does not move the file until the operating system is restarted
|
|
# The system moves the file immediately after AUTOCHK is executed, but before creating any paging files
|
|
$Signature = @{
|
|
Namespace = "WinAPI"
|
|
Name = "DeleteFiles"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
public enum MoveFileFlags
|
|
{
|
|
MOVEFILE_DELAY_UNTIL_REBOOT = 0x00000004
|
|
}
|
|
|
|
[DllImport("kernel32.dll", SetLastError = true, CharSet = CharSet.Unicode)]
|
|
static extern bool MoveFileEx(string lpExistingFileName, string lpNewFileName, MoveFileFlags dwFlags);
|
|
|
|
public static bool MarkFileDelete (string sourcefile)
|
|
{
|
|
return MoveFileEx(sourcefile, null, MoveFileFlags.MOVEFILE_DELAY_UNTIL_REBOOT);
|
|
}
|
|
"@
|
|
}
|
|
|
|
if (-not ("WinAPI.DeleteFiles" -as [type]))
|
|
{
|
|
Add-Type @Signature
|
|
}
|
|
|
|
try
|
|
{
|
|
Remove-Item -Path $env:TEMP -Recurse -Force -ErrorAction Stop
|
|
}
|
|
catch
|
|
{
|
|
# If files are in use remove them at the next boot
|
|
Get-ChildItem -Path $env:TEMP -Recurse -Force -ErrorAction Ignore | ForEach-Object -Process {[WinAPI.DeleteFiles]::MarkFileDelete($_.FullName)}
|
|
}
|
|
|
|
$TempFolder = [System.Environment]::ExpandEnvironmentVariables($env:TEMP)
|
|
$TempFolderCleanupTask = @"
|
|
Remove-Item -Path "$TempFolder" -Recurse -Force
|
|
|
|
Unregister-ScheduledTask -TaskName TemporaryTask -Confirm:`$false
|
|
"@
|
|
|
|
# Create a temporary scheduled task to clean up the temporary folder
|
|
$Action = New-ScheduledTaskAction -Execute powershell.exe -Argument "-WindowStyle Hidden -Command $TempFolderCleanupTask"
|
|
$Trigger = New-ScheduledTaskTrigger -AtLogon -User $env:USERNAME
|
|
$Settings = New-ScheduledTaskSettingsSet -Compatibility Win8
|
|
$Principal = New-ScheduledTaskPrincipal -UserId $env:USERNAME -RunLevel Highest
|
|
$Parameters = @{
|
|
TaskName = "TemporaryTask"
|
|
Principal = $Principal
|
|
Action = $Action
|
|
Settings = $Settings
|
|
Trigger = $Trigger
|
|
}
|
|
Register-ScheduledTask @Parameters -Force
|
|
}
|
|
|
|
#region main
|
|
# Change the %TEMP% environment variable path to %LOCALAPPDATA%\Temp
|
|
[Environment]::SetEnvironmentVariable("TMP", "$env:LOCALAPPDATA\Temp", "User")
|
|
[Environment]::SetEnvironmentVariable("TMP", "$env:SystemRoot\TEMP", "Machine")
|
|
[Environment]::SetEnvironmentVariable("TMP", "$env:LOCALAPPDATA\Temp", "Process")
|
|
New-ItemProperty -Path HKCU:\Environment -Name TMP -PropertyType ExpandString -Value "%USERPROFILE%\AppData\Local\Temp" -Force
|
|
|
|
[Environment]::SetEnvironmentVariable("TEMP", "$env:LOCALAPPDATA\Temp", "User")
|
|
[Environment]::SetEnvironmentVariable("TEMP", "$env:SystemRoot\TEMP", "Machine")
|
|
[Environment]::SetEnvironmentVariable("TEMP", "$env:LOCALAPPDATA\Temp", "Process")
|
|
New-ItemProperty -Path HKCU:\Environment -Name TEMP -PropertyType ExpandString -Value "%USERPROFILE%\AppData\Local\Temp" -Force
|
|
|
|
New-ItemProperty -Path "HKLM:\SYSTEM\CurrentControlSet\Control\Session Manager\Environment" -Name TMP -PropertyType ExpandString -Value "%SystemRoot%\TEMP" -Force
|
|
New-ItemProperty -Path "HKLM:\SYSTEM\CurrentControlSet\Control\Session Manager\Environment" -Name TEMP -PropertyType ExpandString -Value "%SystemRoot%\TEMP" -Force
|
|
# endregion main
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The Windows 260 character path limit
|
|
|
|
.PARAMETER Disable
|
|
Disable the Windows 260 character path limit
|
|
|
|
.PARAMETER Enable
|
|
Enable the Windows 260 character path limit
|
|
|
|
.EXAMPLE
|
|
Win32LongPathLimit -Disable
|
|
|
|
.EXAMPLE
|
|
Win32LongPathLimit -Enable
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function Win32LongPathLimit
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SYSTEM\CurrentControlSet\Control\FileSystem -Name LongPathsEnabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SYSTEM\CurrentControlSet\Control\FileSystem -Name LongPathsEnabled -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Stop error code when BSoD occurs
|
|
|
|
.PARAMETER Enable
|
|
Display Stop error code when BSoD occurs
|
|
|
|
.PARAMETER Disable
|
|
Do not Stop error code when BSoD occurs
|
|
|
|
.EXAMPLE
|
|
BSoDStopError -Enable
|
|
|
|
.EXAMPLE
|
|
BSoDStopError -Disable
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function BSoDStopError
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SYSTEM\CurrentControlSet\Control\CrashControl -Name DisplayParameters -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SYSTEM\CurrentControlSet\Control\CrashControl -Name DisplayParameters -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The User Account Control (UAC) behavior
|
|
|
|
.PARAMETER Never
|
|
Never notify
|
|
|
|
.PARAMETER Default
|
|
Notify me only when apps try to make changes to my computer
|
|
|
|
.EXAMPLE
|
|
AdminApprovalMode -Never
|
|
|
|
.EXAMPLE
|
|
AdminApprovalMode -Default
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function AdminApprovalMode
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Never"
|
|
)]
|
|
[switch]
|
|
$Never,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Default"
|
|
)]
|
|
[switch]
|
|
$Default
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Never"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\System -Name ConsentPromptBehaviorAdmin -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Default"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\System -Name ConsentPromptBehaviorAdmin -PropertyType DWord -Value 5 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Access to mapped drives from app running with elevated permissions with Admin Approval Mode enabled
|
|
|
|
.PARAMETER Enable
|
|
Turn on access to mapped drives from app running with elevated permissions with Admin Approval Mode enabled
|
|
|
|
.PARAMETER Disable
|
|
Turn off access to mapped drives from app running with elevated permissions with Admin Approval Mode enabled
|
|
|
|
.EXAMPLE
|
|
MappedDrivesAppElevatedAccess -Enable
|
|
|
|
.EXAMPLE
|
|
MappedDrivesAppElevatedAccess -Disable
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function MappedDrivesAppElevatedAccess
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\System -Name EnableLinkedConnections -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
Remove-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\System -Name EnableLinkedConnections -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Delivery Optimization
|
|
|
|
.PARAMETER Disable
|
|
Turn off Delivery Optimization
|
|
|
|
.PARAMETER Enable
|
|
Turn on Delivery Optimization
|
|
|
|
.EXAMPLE
|
|
DeliveryOptimization -Disable
|
|
|
|
.EXAMPLE
|
|
DeliveryOptimization -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function DeliveryOptimization
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path Registry::HKEY_USERS\S-1-5-20\SOFTWARE\Microsoft\Windows\CurrentVersion\DeliveryOptimization\Settings -Name DownloadMode -PropertyType DWord -Value 0 -Force
|
|
Delete-DeliveryOptimizationCache -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path Registry::HKEY_USERS\S-1-5-20\SOFTWARE\Microsoft\Windows\CurrentVersion\DeliveryOptimization\Settings -Name DownloadMode -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The Group Policy processing
|
|
|
|
.PARAMETER Enable
|
|
Always wait for the network at computer startup and logon for workgroup networks
|
|
|
|
.PARAMETER Disable
|
|
Never wait for the network at computer startup and logon for workgroup networks
|
|
|
|
.EXAMPLE
|
|
WaitNetworkStartup -Enable
|
|
|
|
.EXAMPLE
|
|
WaitNetworkStartup -Disable
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function WaitNetworkStartup
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
if ((Get-CimInstance -ClassName CIM_ComputerSystem).PartOfDomain)
|
|
{
|
|
if (-not (Test-Path -Path "HKLM:\SOFTWARE\Policies\Microsoft\Windows NT\CurrentVersion\Winlogon"))
|
|
{
|
|
New-Item -Path "HKLM:\SOFTWARE\Policies\Microsoft\Windows NT\CurrentVersion\Winlogon" -Force
|
|
}
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\Policies\Microsoft\Windows NT\CurrentVersion\Winlogon" -Name SyncForegroundPolicy -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
"Disable"
|
|
{
|
|
if ((Get-CimInstance -ClassName CIM_ComputerSystem).PartOfDomain)
|
|
{
|
|
Remove-ItemProperty -Path "HKLM:\SOFTWARE\Policies\Microsoft\Windows NT\CurrentVersion\Winlogon" -Name SyncForegroundPolicy -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Windows manages my default printer
|
|
|
|
.PARAMETER Disable
|
|
Do not let Windows manage my default printer
|
|
|
|
.PARAMETER Enable
|
|
Let Windows manage my default printer
|
|
|
|
.EXAMPLE
|
|
WindowsManageDefaultPrinter -Disable
|
|
|
|
.EXAMPLE
|
|
WindowsManageDefaultPrinter -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function WindowsManageDefaultPrinter
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Windows" -Name LegacyDefaultPrinterMode -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Windows" -Name LegacyDefaultPrinterMode -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Windows features
|
|
|
|
.PARAMETER Disable
|
|
Disable Windows features
|
|
|
|
.PARAMETER Enable
|
|
Enable Windows features
|
|
|
|
.EXAMPLE
|
|
WindowsFeatures -Disable
|
|
|
|
.EXAMPLE
|
|
WindowsFeatures -Enable
|
|
|
|
.NOTES
|
|
A pop-up dialog box lets a user select features
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function WindowsFeatures
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
Add-Type -AssemblyName PresentationCore, PresentationFramework
|
|
|
|
#region Variables
|
|
# Initialize an array list to store the selected Windows features
|
|
$SelectedFeatures = New-Object -TypeName System.Collections.ArrayList($null)
|
|
|
|
# The following Windows features will have their checkboxes checked
|
|
[string[]]$CheckedFeatures = @(
|
|
# Legacy Components
|
|
"LegacyComponents",
|
|
|
|
# PowerShell 2.0
|
|
"MicrosoftWindowsPowerShellV2",
|
|
"MicrosoftWindowsPowershellV2Root",
|
|
|
|
# Microsoft XPS Document Writer
|
|
"Printing-XPSServices-Features",
|
|
|
|
# Work Folders Client
|
|
"WorkFolders-Client"
|
|
)
|
|
|
|
# The following Windows features will have their checkboxes unchecked
|
|
[string[]]$UncheckedFeatures = @(
|
|
# Media Features
|
|
# If you want to leave "Multimedia settings" in the advanced settings of Power Options do not disable this feature
|
|
"MediaPlayback"
|
|
)
|
|
#endregion Variables
|
|
|
|
#region XAML Markup
|
|
# The section defines the design of the upcoming dialog box
|
|
[xml]$XAML = '
|
|
<Window
|
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
|
Name="Window"
|
|
MinHeight="450" MinWidth="400"
|
|
SizeToContent="WidthAndHeight" WindowStartupLocation="CenterScreen"
|
|
TextOptions.TextFormattingMode="Display" SnapsToDevicePixels="True"
|
|
FontFamily="Candara" FontSize="16" ShowInTaskbar="True"
|
|
Background="#F1F1F1" Foreground="#262626">
|
|
<Window.Resources>
|
|
<Style TargetType="StackPanel">
|
|
<Setter Property="Orientation" Value="Horizontal"/>
|
|
<Setter Property="VerticalAlignment" Value="Top"/>
|
|
</Style>
|
|
<Style TargetType="CheckBox">
|
|
<Setter Property="Margin" Value="10, 10, 5, 10"/>
|
|
<Setter Property="IsChecked" Value="True"/>
|
|
</Style>
|
|
<Style TargetType="TextBlock">
|
|
<Setter Property="Margin" Value="5, 10, 10, 10"/>
|
|
</Style>
|
|
<Style TargetType="Button">
|
|
<Setter Property="Margin" Value="20"/>
|
|
<Setter Property="Padding" Value="10"/>
|
|
</Style>
|
|
<Style TargetType="Border">
|
|
<Setter Property="Grid.Row" Value="1"/>
|
|
<Setter Property="CornerRadius" Value="0"/>
|
|
<Setter Property="BorderThickness" Value="0, 1, 0, 1"/>
|
|
<Setter Property="BorderBrush" Value="#000000"/>
|
|
</Style>
|
|
<Style TargetType="ScrollViewer">
|
|
<Setter Property="HorizontalScrollBarVisibility" Value="Disabled"/>
|
|
<Setter Property="BorderBrush" Value="#000000"/>
|
|
<Setter Property="BorderThickness" Value="0, 1, 0, 1"/>
|
|
</Style>
|
|
</Window.Resources>
|
|
<Grid>
|
|
<Grid.RowDefinitions>
|
|
<RowDefinition Height="Auto"/>
|
|
<RowDefinition Height="*"/>
|
|
<RowDefinition Height="Auto"/>
|
|
</Grid.RowDefinitions>
|
|
<ScrollViewer Name="Scroll" Grid.Row="0"
|
|
HorizontalScrollBarVisibility="Disabled"
|
|
VerticalScrollBarVisibility="Auto">
|
|
<StackPanel Name="PanelContainer" Orientation="Vertical"/>
|
|
</ScrollViewer>
|
|
<Button Name="Button" Grid.Row="2"/>
|
|
</Grid>
|
|
</Window>
|
|
'
|
|
#endregion XAML Markup
|
|
|
|
$Reader = (New-Object -TypeName System.Xml.XmlNodeReader -ArgumentList $XAML)
|
|
$Form = [Windows.Markup.XamlReader]::Load($Reader)
|
|
$XAML.SelectNodes("//*[@*[contains(translate(name(.),'n','N'),'Name')]]") | ForEach-Object -Process {
|
|
Set-Variable -Name ($_.Name) -Value $Form.FindName($_.Name)
|
|
}
|
|
|
|
#region Functions
|
|
function Get-CheckboxClicked
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ValueFromPipeline = $true
|
|
)]
|
|
[ValidateNotNull()]
|
|
$CheckBox
|
|
)
|
|
|
|
$Feature = $Features | Where-Object -FilterScript {$_.DisplayName -eq $CheckBox.Parent.Children[1].Text}
|
|
|
|
if ($CheckBox.IsChecked)
|
|
{
|
|
[void]$SelectedFeatures.Add($Feature)
|
|
}
|
|
else
|
|
{
|
|
[void]$SelectedFeatures.Remove($Feature)
|
|
}
|
|
if ($SelectedFeatures.Count -gt 0)
|
|
{
|
|
$Button.IsEnabled = $true
|
|
}
|
|
else
|
|
{
|
|
$Button.IsEnabled = $false
|
|
}
|
|
}
|
|
|
|
function DisableButton
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
[void]$Window.Close()
|
|
|
|
$SelectedFeatures | ForEach-Object -Process {Write-Verbose $_.DisplayName -Verbose}
|
|
$SelectedFeatures | Disable-WindowsOptionalFeature -Online -NoRestart
|
|
}
|
|
|
|
function EnableButton
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
[void]$Window.Close()
|
|
|
|
$SelectedFeatures | ForEach-Object -Process {Write-Verbose -Message $_.DisplayName -Verbose}
|
|
$SelectedFeatures | Enable-WindowsOptionalFeature -Online -NoRestart
|
|
}
|
|
|
|
function Add-FeatureControl
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ValueFromPipeline = $true
|
|
)]
|
|
[ValidateNotNull()]
|
|
$Feature
|
|
)
|
|
|
|
process
|
|
{
|
|
$CheckBox = New-Object -TypeName System.Windows.Controls.CheckBox
|
|
$CheckBox.Add_Click({Get-CheckboxClicked -CheckBox $_.Source})
|
|
$CheckBox.ToolTip = $Feature.Description
|
|
|
|
$TextBlock = New-Object -TypeName System.Windows.Controls.TextBlock
|
|
$TextBlock.Text = $Feature.DisplayName
|
|
$TextBlock.ToolTip = $Feature.Description
|
|
|
|
$StackPanel = New-Object -TypeName System.Windows.Controls.StackPanel
|
|
[void]$StackPanel.Children.Add($CheckBox)
|
|
[void]$StackPanel.Children.Add($TextBlock)
|
|
[void]$PanelContainer.Children.Add($StackPanel)
|
|
|
|
$CheckBox.IsChecked = $true
|
|
|
|
# If feature checked add to the array list
|
|
if ($UnCheckedFeatures | Where-Object -FilterScript {$Feature.FeatureName -like $_})
|
|
{
|
|
$CheckBox.IsChecked = $false
|
|
# Exit function if item is not checked
|
|
return
|
|
}
|
|
|
|
# If feature checked add to the array list
|
|
[void]$SelectedFeatures.Add($Feature)
|
|
}
|
|
}
|
|
#endregion Functions
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
$State = @("Disabled", "DisablePending")
|
|
$ButtonContent = $Localization.Enable
|
|
$ButtonAdd_Click = {EnableButton}
|
|
}
|
|
"Disable"
|
|
{
|
|
$State = @("Enabled", "EnablePending")
|
|
$ButtonContent = $Localization.Disable
|
|
$ButtonAdd_Click = {DisableButton}
|
|
}
|
|
}
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
# Getting list of all optional features according to the conditions
|
|
$OFS = "|"
|
|
$Features = Get-WindowsOptionalFeature -Online | Where-Object -FilterScript {
|
|
($_.State -in $State) -and (($_.FeatureName -match $UncheckedFeatures) -or ($_.FeatureName -match $CheckedFeatures))
|
|
} | ForEach-Object -Process {Get-WindowsOptionalFeature -FeatureName $_.FeatureName -Online}
|
|
$OFS = " "
|
|
|
|
if (-not ($Features))
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.NoData -Verbose
|
|
|
|
return
|
|
}
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.DialogBoxOpening -Verbose
|
|
|
|
#region Sendkey function
|
|
# Emulate the Backspace key sending to prevent the console window to freeze
|
|
Start-Sleep -Milliseconds 500
|
|
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
|
|
$SetForegroundWindow = @{
|
|
Namespace = "WinAPI"
|
|
Name = "ForegroundWindow"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
[DllImport("user32.dll")]
|
|
public static extern bool ShowWindowAsync(IntPtr hWnd, int nCmdShow);
|
|
|
|
[DllImport("user32.dll")]
|
|
[return: MarshalAs(UnmanagedType.Bool)]
|
|
public static extern bool SetForegroundWindow(IntPtr hWnd);
|
|
"@
|
|
}
|
|
|
|
if (-not ("WinAPI.ForegroundWindow" -as [type]))
|
|
{
|
|
Add-Type @SetForegroundWindow
|
|
}
|
|
|
|
Get-Process | Where-Object -FilterScript {(($_.ProcessName -eq "powershell") -or ($_.ProcessName -eq "WindowsTerminal")) -and ($_.MainWindowTitle -match "Sophia Script for Windows 11")} | ForEach-Object -Process {
|
|
# Show window, if minimized
|
|
[WinAPI.ForegroundWindow]::ShowWindowAsync($_.MainWindowHandle, 10)
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Force move the console window to the foreground
|
|
[WinAPI.ForegroundWindow]::SetForegroundWindow($_.MainWindowHandle)
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Emulate the Backspace key sending
|
|
[System.Windows.Forms.SendKeys]::SendWait("{BACKSPACE 1}")
|
|
}
|
|
#endregion Sendkey function
|
|
|
|
$Window.Add_Loaded({$Features | Add-FeatureControl})
|
|
$Button.Content = $ButtonContent
|
|
$Button.Add_Click({& $ButtonAdd_Click})
|
|
|
|
$Window.Title = $Localization.WindowsFeaturesTitle
|
|
|
|
# Force move the WPF form to the foreground
|
|
$Window.Add_Loaded({$Window.Activate()})
|
|
$Form.ShowDialog() | Out-Null
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Optional features
|
|
|
|
.PARAMETER Uninstall
|
|
Uninstall optional features
|
|
|
|
.PARAMETER Install
|
|
Install optional features
|
|
|
|
.EXAMPLE
|
|
WindowsCapabilities -Uninstall
|
|
|
|
.EXAMPLE
|
|
WindowsCapabilities -Install
|
|
|
|
.NOTES
|
|
A pop-up dialog box lets a user select features
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function WindowsCapabilities
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Uninstall"
|
|
)]
|
|
[switch]
|
|
$Uninstall,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Install"
|
|
)]
|
|
[switch]
|
|
$Install
|
|
)
|
|
|
|
Add-Type -AssemblyName PresentationCore, PresentationFramework
|
|
|
|
#region Variables
|
|
# Initialize an array list to store the selected optional features
|
|
$SelectedCapabilities = New-Object -TypeName System.Collections.ArrayList($null)
|
|
|
|
# The following optional features will have their checkboxes checked
|
|
[string[]]$CheckedCapabilities = @(
|
|
# Steps Recorder
|
|
"App.StepsRecorder*",
|
|
|
|
# Microsoft Quick Assist
|
|
"App.Support.QuickAssist*",
|
|
|
|
# WordPad
|
|
"Microsoft.Windows.WordPad*"
|
|
)
|
|
|
|
# The following optional features will have their checkboxes unchecked
|
|
[string[]]$UncheckedCapabilities = @(
|
|
# Internet Explorer mode
|
|
"Browser.InternetExplorer*",
|
|
|
|
# Math Recognizer
|
|
"MathRecognizer*",
|
|
|
|
# Windows Media Player
|
|
# If you want to leave "Multimedia settings" element in the advanced settings of Power Options do not uninstall this feature
|
|
"Media.WindowsMediaPlayer*",
|
|
|
|
# OpenSSH Client
|
|
"OpenSSH.Client*"
|
|
)
|
|
|
|
# The following optional features will be excluded from the display
|
|
[string[]]$ExcludedCapabilities = @(
|
|
# The DirectX Database to configure and optimize apps when multiple Graphics Adapters are present
|
|
"DirectX.Configuration.Database*",
|
|
|
|
# Language components
|
|
"Language.*",
|
|
|
|
# Notepad
|
|
"Microsoft.Windows.Notepad*",
|
|
|
|
# Mail, contacts, and calendar sync component
|
|
"OneCoreUAP.OneSync*",
|
|
|
|
# Windows PowerShell Intergrated Scripting Enviroment
|
|
"Microsoft.Windows.PowerShell.ISE*",
|
|
|
|
# Management of printers, printer drivers, and printer servers
|
|
"Print.Management.Console*",
|
|
|
|
# Features critical to Windows functionality
|
|
"Windows.Client.ShellComponents*"
|
|
)
|
|
#endregion Variables
|
|
|
|
#region XAML Markup
|
|
# The section defines the design of the upcoming dialog box
|
|
[xml]$XAML = '
|
|
<Window
|
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
|
Name="Window"
|
|
MinHeight="450" MinWidth="400"
|
|
SizeToContent="WidthAndHeight" WindowStartupLocation="CenterScreen"
|
|
TextOptions.TextFormattingMode="Display" SnapsToDevicePixels="True"
|
|
FontFamily="Candara" FontSize="16" ShowInTaskbar="True"
|
|
Background="#F1F1F1" Foreground="#262626">
|
|
<Window.Resources>
|
|
<Style TargetType="StackPanel">
|
|
<Setter Property="Orientation" Value="Horizontal"/>
|
|
<Setter Property="VerticalAlignment" Value="Top"/>
|
|
</Style>
|
|
<Style TargetType="CheckBox">
|
|
<Setter Property="Margin" Value="10, 10, 5, 10"/>
|
|
<Setter Property="IsChecked" Value="True"/>
|
|
</Style>
|
|
<Style TargetType="TextBlock">
|
|
<Setter Property="Margin" Value="5, 10, 10, 10"/>
|
|
</Style>
|
|
<Style TargetType="Button">
|
|
<Setter Property="Margin" Value="20"/>
|
|
<Setter Property="Padding" Value="10"/>
|
|
</Style>
|
|
<Style TargetType="Border">
|
|
<Setter Property="Grid.Row" Value="1"/>
|
|
<Setter Property="CornerRadius" Value="0"/>
|
|
<Setter Property="BorderThickness" Value="0, 1, 0, 1"/>
|
|
<Setter Property="BorderBrush" Value="#000000"/>
|
|
</Style>
|
|
<Style TargetType="ScrollViewer">
|
|
<Setter Property="HorizontalScrollBarVisibility" Value="Disabled"/>
|
|
<Setter Property="BorderBrush" Value="#000000"/>
|
|
<Setter Property="BorderThickness" Value="0, 1, 0, 1"/>
|
|
</Style>
|
|
</Window.Resources>
|
|
<Grid>
|
|
<Grid.RowDefinitions>
|
|
<RowDefinition Height="Auto"/>
|
|
<RowDefinition Height="*"/>
|
|
<RowDefinition Height="Auto"/>
|
|
</Grid.RowDefinitions>
|
|
<ScrollViewer Name="Scroll" Grid.Row="0"
|
|
HorizontalScrollBarVisibility="Disabled"
|
|
VerticalScrollBarVisibility="Auto">
|
|
<StackPanel Name="PanelContainer" Orientation="Vertical"/>
|
|
</ScrollViewer>
|
|
<Button Name="Button" Grid.Row="2"/>
|
|
</Grid>
|
|
</Window>
|
|
'
|
|
#endregion XAML Markup
|
|
|
|
$Reader = (New-Object -TypeName System.Xml.XmlNodeReader -ArgumentList $XAML)
|
|
$Form = [Windows.Markup.XamlReader]::Load($Reader)
|
|
$XAML.SelectNodes("//*[@*[contains(translate(name(.),'n','N'),'Name')]]") | ForEach-Object -Process {
|
|
Set-Variable -Name ($_.Name) -Value $Form.FindName($_.Name)
|
|
}
|
|
|
|
#region Functions
|
|
function Get-CheckboxClicked
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ValueFromPipeline = $true
|
|
)]
|
|
[ValidateNotNull()]
|
|
$CheckBox
|
|
)
|
|
|
|
$Capability = $Capabilities | Where-Object -FilterScript {$_.DisplayName -eq $CheckBox.Parent.Children[1].Text}
|
|
|
|
if ($CheckBox.IsChecked)
|
|
{
|
|
[void]$SelectedCapabilities.Add($Capability)
|
|
}
|
|
else
|
|
{
|
|
[void]$SelectedCapabilities.Remove($Capability)
|
|
}
|
|
|
|
if ($SelectedCapabilities.Count -gt 0)
|
|
{
|
|
$Button.IsEnabled = $true
|
|
}
|
|
else
|
|
{
|
|
$Button.IsEnabled = $false
|
|
}
|
|
}
|
|
|
|
function UninstallButton
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
[void]$Window.Close()
|
|
|
|
$SelectedCapabilities | ForEach-Object -Process {Write-Verbose -Message $_.DisplayName -Verbose}
|
|
$SelectedCapabilities | Where-Object -FilterScript {$_.Name -in (Get-WindowsCapability -Online).Name} | Remove-WindowsCapability -Online
|
|
|
|
if ([string]$SelectedCapabilities.Name -match "Browser.InternetExplorer")
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.RestartWarning
|
|
}
|
|
}
|
|
|
|
function InstallButton
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
[void]$Window.Close()
|
|
|
|
$SelectedCapabilities | ForEach-Object -Process {Write-Verbose -Message $_.DisplayName -Verbose}
|
|
$SelectedCapabilities | Where-Object -FilterScript {$_.Name -in ((Get-WindowsCapability -Online).Name)} | Add-WindowsCapability -Online
|
|
|
|
if ([string]$SelectedCapabilities.Name -match "Browser.InternetExplorer")
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.RestartWarning
|
|
}
|
|
}
|
|
|
|
function Add-CapabilityControl
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ValueFromPipeline = $true
|
|
)]
|
|
[ValidateNotNull()]
|
|
$Capability
|
|
)
|
|
|
|
process
|
|
{
|
|
$CheckBox = New-Object -TypeName System.Windows.Controls.CheckBox
|
|
$CheckBox.Add_Click({Get-CheckboxClicked -CheckBox $_.Source})
|
|
$CheckBox.ToolTip = $Capability.Description
|
|
|
|
$TextBlock = New-Object -TypeName System.Windows.Controls.TextBlock
|
|
$TextBlock.Text = $Capability.DisplayName
|
|
$TextBlock.ToolTip = $Capability.Description
|
|
|
|
$StackPanel = New-Object -TypeName System.Windows.Controls.StackPanel
|
|
[void]$StackPanel.Children.Add($CheckBox)
|
|
[void]$StackPanel.Children.Add($TextBlock)
|
|
[void]$PanelContainer.Children.Add($StackPanel)
|
|
|
|
# If capability checked add to the array list
|
|
if ($UnCheckedCapabilities | Where-Object -FilterScript {$Capability.Name -like $_})
|
|
{
|
|
$CheckBox.IsChecked = $false
|
|
# Exit function if item is not checked
|
|
return
|
|
}
|
|
|
|
# If capability checked add to the array list
|
|
[void]$SelectedCapabilities.Add($Capability)
|
|
}
|
|
}
|
|
#endregion Functions
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Install"
|
|
{
|
|
try
|
|
{
|
|
# Check the internet connection
|
|
$Parameters = @{
|
|
Uri = "https://www.google.com"
|
|
Method = "Head"
|
|
DisableKeepAlive = $true
|
|
UseBasicParsing = $true
|
|
}
|
|
if (-not (Invoke-WebRequest @Parameters).StatusDescription)
|
|
{
|
|
return
|
|
}
|
|
|
|
$State = "NotPresent"
|
|
$ButtonContent = $Localization.Install
|
|
$ButtonAdd_Click = {InstallButton}
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message $Localization.NoInternetConnection
|
|
Write-Error -Message $Localization.NoInternetConnection -ErrorAction SilentlyContinue
|
|
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
"Uninstall"
|
|
{
|
|
$State = "Installed"
|
|
$ButtonContent = $Localization.Uninstall
|
|
$ButtonAdd_Click = {UninstallButton}
|
|
}
|
|
}
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
# Getting list of all capabilities according to the conditions
|
|
$OFS = "|"
|
|
$Capabilities = Get-WindowsCapability -Online | Where-Object -FilterScript {
|
|
($_.State -eq $State) -and (($_.Name -match $UncheckedCapabilities) -or ($_.Name -match $CheckedCapabilities) -and ($_.Name -notmatch $ExcludedCapabilities))
|
|
} | ForEach-Object -Process {Get-WindowsCapability -Name $_.Name -Online}
|
|
$OFS = " "
|
|
|
|
if (-not ($Capabilities))
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.NoData -Verbose
|
|
|
|
return
|
|
}
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.DialogBoxOpening -Verbose
|
|
|
|
#region Sendkey function
|
|
# Emulate the Backspace key sending to prevent the console window to freeze
|
|
Start-Sleep -Milliseconds 500
|
|
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
|
|
$SetForegroundWindow = @{
|
|
Namespace = "WinAPI"
|
|
Name = "ForegroundWindow"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
[DllImport("user32.dll")]
|
|
public static extern bool ShowWindowAsync(IntPtr hWnd, int nCmdShow);
|
|
|
|
[DllImport("user32.dll")]
|
|
[return: MarshalAs(UnmanagedType.Bool)]
|
|
public static extern bool SetForegroundWindow(IntPtr hWnd);
|
|
"@
|
|
}
|
|
|
|
if (-not ("WinAPI.ForegroundWindow" -as [type]))
|
|
{
|
|
Add-Type @SetForegroundWindow
|
|
}
|
|
|
|
Get-Process | Where-Object -FilterScript {(($_.ProcessName -eq "powershell") -or ($_.ProcessName -eq "WindowsTerminal")) -and ($_.MainWindowTitle -match "Sophia Script for Windows 11")} | ForEach-Object -Process {
|
|
# Show window, if minimized
|
|
[WinAPI.ForegroundWindow]::ShowWindowAsync($_.MainWindowHandle, 10)
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Force move the console window to the foreground
|
|
[WinAPI.ForegroundWindow]::SetForegroundWindow($_.MainWindowHandle)
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Emulate the Backspace key sending
|
|
[System.Windows.Forms.SendKeys]::SendWait("{BACKSPACE 1}")
|
|
}
|
|
#endregion Sendkey function
|
|
|
|
$Window.Add_Loaded({$Capabilities | Add-CapabilityControl})
|
|
$Button.Content = $ButtonContent
|
|
$Button.Add_Click({& $ButtonAdd_Click})
|
|
|
|
$Window.Title = $Localization.OptionalFeaturesTitle
|
|
|
|
# Force move the WPF form to the foreground
|
|
$Window.Add_Loaded({$Window.Activate()})
|
|
$Form.ShowDialog() | Out-Null
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Receive updates for other Microsoft products
|
|
|
|
.PARAMETER Enable
|
|
Receive updates for other Microsoft products
|
|
|
|
.PARAMETER Disable
|
|
Do not receive updates for other Microsoft products
|
|
|
|
.EXAMPLE
|
|
UpdateMicrosoftProducts -Enable
|
|
|
|
.EXAMPLE
|
|
UpdateMicrosoftProducts -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function UpdateMicrosoftProducts
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
(New-Object -ComObject Microsoft.Update.ServiceManager).AddService2("7971f918-a847-4430-9279-4a52d1efe18d", 7, "")
|
|
}
|
|
"Disable"
|
|
{
|
|
if (((New-Object -ComObject Microsoft.Update.ServiceManager).Services | Where-Object -FilterScript {$_.ServiceID -eq "7971f918-a847-4430-9279-4a52d1efe18d"}).IsDefaultAUService)
|
|
{
|
|
(New-Object -ComObject Microsoft.Update.ServiceManager).RemoveService("7971f918-a847-4430-9279-4a52d1efe18d")
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Power plan
|
|
|
|
.PARAMETER High
|
|
Set power plan on "High performance"
|
|
|
|
.PARAMETER Balanced
|
|
Set power plan on "Balanced"
|
|
|
|
.EXAMPLE
|
|
PowerPlan -High
|
|
|
|
.EXAMPLE
|
|
PowerPlan -Balanced
|
|
|
|
.NOTES
|
|
It isn't recommended to turn on the "High performance" power plan on laptops
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function PowerPlan
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "High"
|
|
)]
|
|
[switch]
|
|
$High,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Balanced"
|
|
)]
|
|
[switch]
|
|
$Balanced
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"High"
|
|
{
|
|
POWERCFG /SETACTIVE SCHEME_MIN
|
|
}
|
|
"Balanced"
|
|
{
|
|
POWERCFG /SETACTIVE SCHEME_BALANCED
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The the latest installed .NET runtime for all apps usage
|
|
|
|
.PARAMETER Enable
|
|
Use the latest installed .NET runtime for all apps
|
|
|
|
.PARAMETER Disable
|
|
Do not use the latest installed .NET runtime for all apps
|
|
|
|
.EXAMPLE
|
|
LatestInstalled.NET -Enable
|
|
|
|
.EXAMPLE
|
|
LatestInstalled.NET -Disable
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function LatestInstalled.NET
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\.NETFramework -Name OnlyUseLatestCLR -PropertyType DWord -Value 1 -Force
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Wow6432Node\Microsoft\.NETFramework -Name OnlyUseLatestCLR -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
Remove-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\.NETFramework -Name OnlyUseLatestCLR -Force -ErrorAction Ignore
|
|
Remove-ItemProperty -Path HKLM:\SOFTWARE\Wow6432Node\Microsoft\.NETFramework -Name OnlyUseLatestCLR -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Network adapters power management
|
|
|
|
.PARAMETER Disable
|
|
Do not allow the computer to turn off the network adapters to save power
|
|
|
|
.PARAMETER Enable
|
|
Allow the computer to turn off the network adapters to save power
|
|
|
|
.EXAMPLE
|
|
NetworkAdaptersSavePower -Disable
|
|
|
|
.EXAMPLE
|
|
NetworkAdaptersSavePower -Enable
|
|
|
|
.NOTES
|
|
Do not recommend turning it on on laptops
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function NetworkAdaptersSavePower
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
if (Get-NetAdapter -Physical | Where-Object -FilterScript {$_.Status -eq "Up"})
|
|
{
|
|
$PhysicalAdaptersStatusUp = @((Get-NetAdapter -Physical | Where-Object -FilterScript {$_.Status -eq "Up"}).Name)
|
|
}
|
|
|
|
$Adapters = Get-NetAdapter -Physical | Get-NetAdapterPowerManagement | Where-Object -FilterScript {$_.AllowComputerToTurnOffDevice -ne "Unsupported"}
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
foreach ($Adapter in $Adapters)
|
|
{
|
|
$Adapter.AllowComputerToTurnOffDevice = "Disabled"
|
|
$Adapter | Set-NetAdapterPowerManagement
|
|
}
|
|
}
|
|
"Enable"
|
|
{
|
|
foreach ($Adapter in $Adapters)
|
|
{
|
|
$Adapter.AllowComputerToTurnOffDevice = "Enabled"
|
|
$Adapter | Set-NetAdapterPowerManagement
|
|
}
|
|
}
|
|
}
|
|
|
|
# All network adapters are turned into "Disconnected" for few seconds, so we need to wait a bit to let them up
|
|
# Otherwise functions below will indicate that there is no the Internet connection
|
|
if ($PhysicalAdaptersStatusUp)
|
|
{
|
|
while
|
|
(
|
|
Get-NetAdapter -Physical -Name $PhysicalAdaptersStatusUp | ForEach-Object -Process {$_.Status -eq "Disconnected"}
|
|
)
|
|
{
|
|
Start-Sleep -Seconds 2
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Internet Protocol Version 6 (TCP/IPv6) component
|
|
|
|
.PARAMETER Disable
|
|
Disable the Internet Protocol Version 6 (TCP/IPv6) component for all network connections if your ISP doesn't support it
|
|
|
|
.PARAMETER Enable
|
|
Enable the Internet Protocol Version 6 (TCP/IPv6) component for all network connections if your ISP supports it
|
|
|
|
.EXAMPLE
|
|
IPv6Component -Disable
|
|
|
|
.EXAMPLE
|
|
IPv6Component -Enable
|
|
|
|
.NOTES
|
|
Before invoking the function, a check will be run whether your ISP supports the IPv6 protocol using https://ipv6-test.com
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function IPv6Component
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
try
|
|
{
|
|
# Check the internet connection
|
|
$Parameters = @{
|
|
Uri = "https://www.google.com"
|
|
Method = "Head"
|
|
DisableKeepAlive = $true
|
|
UseBasicParsing = $true
|
|
}
|
|
if (-not (Invoke-WebRequest @Parameters).StatusDescription)
|
|
{
|
|
return
|
|
}
|
|
|
|
try
|
|
{
|
|
# Check whether the https://ipv6-test.com site is alive
|
|
$Parameters = @{
|
|
Uri = "https://ipv6-test.com"
|
|
Method = "Head"
|
|
DisableKeepAlive = $true
|
|
UseBasicParsing = $true
|
|
}
|
|
if (-not (Invoke-WebRequest @Parameters).StatusDescription)
|
|
{
|
|
return
|
|
}
|
|
|
|
# Check whether the ISP supports IPv6 protocol using https://ipv6-test.com
|
|
$Parameters = @{
|
|
Uri = "https://v4v6.ipv6-test.com/api/myip.php?json"
|
|
UseBasicParsing = $true
|
|
}
|
|
$IPVersion = (Invoke-RestMethod @Parameters).proto
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message ($Localization.NoResponse -f "https://ipv6-test.com")
|
|
Write-Error -Message ($Localization.NoResponse -f "https://ipv6-test.com") -ErrorAction SilentlyContinue
|
|
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message $Localization.NoInternetConnection
|
|
Write-Error -Message $Localization.NoInternetConnection -ErrorAction SilentlyContinue
|
|
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if ($IPVersion -ne "ipv6")
|
|
{
|
|
Disable-NetAdapterBinding -Name * -ComponentID ms_tcpip6
|
|
}
|
|
}
|
|
"Enable"
|
|
{
|
|
if ($IPVersion -eq "ipv6")
|
|
{
|
|
Enable-NetAdapterBinding -Name * -ComponentID ms_tcpip6
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Override for default input method
|
|
|
|
.PARAMETER English
|
|
Override for default input method: English
|
|
|
|
.PARAMETER Default
|
|
Override for default input method: use language list
|
|
|
|
.EXAMPLE
|
|
InputMethod -English
|
|
|
|
.EXAMPLE
|
|
InputMethod -Default
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function InputMethod
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "English"
|
|
)]
|
|
[switch]
|
|
$English,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Default"
|
|
)]
|
|
[switch]
|
|
$Default
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"English"
|
|
{
|
|
Set-WinDefaultInputMethodOverride -InputTip "0409:00000409"
|
|
}
|
|
"Default"
|
|
{
|
|
Remove-ItemProperty -Path "HKCU:\Control Panel\International\User Profile" -Name InputMethodOverride -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
User folders location
|
|
|
|
.PARAMETER Root
|
|
Move user folders location to the root of any drive using the interactive menu
|
|
|
|
.PARAMETER Custom
|
|
Select folders for user folders location manually using a folder browser dialog
|
|
|
|
.PARAMETER Default
|
|
Change user folders location to the default values
|
|
|
|
.EXAMPLE
|
|
SetUserShellFolderLocation -Root
|
|
|
|
.EXAMPLE
|
|
SetUserShellFolderLocation -Custom
|
|
|
|
.EXAMPLE
|
|
SetUserShellFolderLocation -Default
|
|
|
|
.NOTES
|
|
User files or folders won't me moved to a new location
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function SetUserShellFolderLocation
|
|
{
|
|
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Root"
|
|
)]
|
|
[switch]
|
|
$Root,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Custom"
|
|
)]
|
|
[switch]
|
|
$Custom,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Default"
|
|
)]
|
|
[switch]
|
|
$Default
|
|
)
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Change the location of the each user folder using SHSetKnownFolderPath function
|
|
|
|
.PARAMETER RemoveDesktopINI
|
|
The RemoveDesktopINI argument removes desktop.ini in the old user shell folder
|
|
|
|
.EXAMPLE
|
|
UserShellFolder -UserFolder Desktop -FolderPath "$env:SystemDrive:\Desktop" -RemoveDesktopINI
|
|
|
|
.LINK
|
|
https://docs.microsoft.com/en-us/windows/win32/api/shlobj_core/nf-shlobj_core-shgetknownfolderpath
|
|
|
|
.NOTES
|
|
User files or folders won't me moved to a new location
|
|
#>
|
|
function UserShellFolder
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(Mandatory = $true)]
|
|
[ValidateSet("Desktop", "Documents", "Downloads", "Music", "Pictures", "Videos")]
|
|
[string]
|
|
$UserFolder,
|
|
|
|
[Parameter(Mandatory = $true)]
|
|
[string]
|
|
$FolderPath,
|
|
|
|
[Parameter(Mandatory = $false)]
|
|
[switch]
|
|
$RemoveDesktopINI
|
|
)
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Redirect user folders to a new location
|
|
|
|
.EXAMPLE
|
|
KnownFolderPath -KnownFolder Desktop -Path "$env:SystemDrive:\Desktop"
|
|
#>
|
|
function KnownFolderPath
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(Mandatory = $true)]
|
|
[ValidateSet("Desktop", "Documents", "Downloads", "Music", "Pictures", "Videos")]
|
|
[string]
|
|
$KnownFolder,
|
|
|
|
[Parameter(Mandatory = $true)]
|
|
[string]
|
|
$Path
|
|
)
|
|
|
|
$KnownFolders = @{
|
|
"Desktop" = @("B4BFCC3A-DB2C-424C-B029-7FE99A87C641");
|
|
"Documents" = @("FDD39AD0-238F-46AF-ADB4-6C85480369C7", "f42ee2d3-909f-4907-8871-4c22fc0bf756");
|
|
"Downloads" = @("374DE290-123F-4565-9164-39C4925E467B", "7d83ee9b-2244-4e70-b1f5-5393042af1e4");
|
|
"Music" = @("4BD8D571-6D19-48D3-BE97-422220080E43", "a0c69a99-21c8-4671-8703-7934162fcf1d");
|
|
"Pictures" = @("33E28130-4E1E-4676-835A-98395C3BC3BB", "0ddd015d-b06c-45d5-8c4c-f59713854639");
|
|
"Videos" = @("18989B1D-99B5-455B-841C-AB7C74E4DDFC", "35286a68-3c57-41a1-bbb1-0eae73d76c95");
|
|
}
|
|
|
|
$Signature = @{
|
|
Namespace = "WinAPI"
|
|
Name = "KnownFolders"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
[DllImport("shell32.dll")]
|
|
public extern static int SHSetKnownFolderPath(ref Guid folderId, uint flags, IntPtr token, [MarshalAs(UnmanagedType.LPWStr)] string path);
|
|
"@
|
|
}
|
|
if (-not ("WinAPI.KnownFolders" -as [type]))
|
|
{
|
|
Add-Type @Signature
|
|
}
|
|
|
|
foreach ($GUID in $KnownFolders[$KnownFolder])
|
|
{
|
|
[WinAPI.KnownFolders]::SHSetKnownFolderPath([ref]$GUID, 0, 0, $Path)
|
|
}
|
|
(Get-Item -Path $Path -Force).Attributes = "ReadOnly"
|
|
}
|
|
|
|
$UserShellFoldersRegistryNames = @{
|
|
"Desktop" = "Desktop"
|
|
"Documents" = "Personal"
|
|
"Downloads" = "{374DE290-123F-4565-9164-39C4925E467B}"
|
|
"Music" = "My Music"
|
|
"Pictures" = "My Pictures"
|
|
"Videos" = "My Video"
|
|
}
|
|
|
|
$UserShellFoldersGUIDs = @{
|
|
"Desktop" = "{754AC886-DF64-4CBA-86B5-F7FBF4FBCEF5}"
|
|
"Documents" = "{F42EE2D3-909F-4907-8871-4C22FC0BF756}"
|
|
"Downloads" = "{7D83EE9B-2244-4E70-B1F5-5393042AF1E4}"
|
|
"Music" = "{A0C69A99-21C8-4671-8703-7934162FCF1D}"
|
|
"Pictures" = "{0DDD015D-B06C-45D5-8C4C-F59713854639}"
|
|
"Videos" = "{35286A68-3C57-41A1-BBB1-0EAE73D76C95}"
|
|
}
|
|
|
|
# Contents of the hidden desktop.ini file for each type of user folders
|
|
$DesktopINI = @{
|
|
"Desktop" = "",
|
|
"[.ShellClassInfo]",
|
|
"LocalizedResourceName=@%SystemRoot%\system32\shell32.dll,-21769",
|
|
"IconResource=%SystemRoot%\system32\imageres.dll,-183"
|
|
"Documents" = "",
|
|
"[.ShellClassInfo]",
|
|
"LocalizedResourceName=@%SystemRoot%\system32\shell32.dll,-21770",
|
|
"IconResource=%SystemRoot%\system32\imageres.dll,-112",
|
|
"IconFile=%SystemRoot%\system32\shell32.dll",
|
|
"IconIndex=-235"
|
|
"Downloads" = "",
|
|
"[.ShellClassInfo]","LocalizedResourceName=@%SystemRoot%\system32\shell32.dll,-21798",
|
|
"IconResource=%SystemRoot%\system32\imageres.dll,-184"
|
|
"Music" = "",
|
|
"[.ShellClassInfo]","LocalizedResourceName=@%SystemRoot%\system32\shell32.dll,-21790",
|
|
"InfoTip=@%SystemRoot%\system32\shell32.dll,-12689",
|
|
"IconResource=%SystemRoot%\system32\imageres.dll,-108",
|
|
"IconFile=%SystemRoot%\system32\shell32.dll","IconIndex=-237"
|
|
"Pictures" = "",
|
|
"[.ShellClassInfo]",
|
|
"LocalizedResourceName=@%SystemRoot%\system32\shell32.dll,-21779",
|
|
"InfoTip=@%SystemRoot%\system32\shell32.dll,-12688",
|
|
"IconResource=%SystemRoot%\system32\imageres.dll,-113",
|
|
"IconFile=%SystemRoot%\system32\shell32.dll",
|
|
"IconIndex=-236"
|
|
"Videos" = "",
|
|
"[.ShellClassInfo]",
|
|
"LocalizedResourceName=@%SystemRoot%\system32\shell32.dll,-21791",
|
|
"InfoTip=@%SystemRoot%\system32\shell32.dll,-12690",
|
|
"IconResource=%SystemRoot%\system32\imageres.dll,-189",
|
|
"IconFile=%SystemRoot%\system32\shell32.dll","IconIndex=-238"
|
|
}
|
|
|
|
# Determining the current user folder path
|
|
$CurrentUserFolderPath = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name $UserShellFoldersRegistryNames[$UserFolder]
|
|
if ($CurrentUserFolder -ne $FolderPath)
|
|
{
|
|
if ((Get-ChildItem -Path $CurrentUserFolderPath | Measure-Object).Count -ne 0)
|
|
{
|
|
Write-Error -Message ($Localization.UserShellFolderNotEmpty -f $CurrentUserFolderPath) -ErrorAction SilentlyContinue
|
|
}
|
|
|
|
# Creating a new folder if there is no one
|
|
if (-not (Test-Path -Path $FolderPath))
|
|
{
|
|
New-Item -Path $FolderPath -ItemType Directory -Force
|
|
}
|
|
|
|
# Removing old desktop.ini
|
|
if ($RemoveDesktopINI.IsPresent)
|
|
{
|
|
Remove-Item -Path "$CurrentUserFolderPath\desktop.ini" -Force
|
|
}
|
|
|
|
KnownFolderPath -KnownFolder $UserFolder -Path $FolderPath
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name $UserShellFoldersGUIDs[$UserFolder] -PropertyType ExpandString -Value $FolderPath -Force
|
|
|
|
# Save desktop.ini in the UTF-16 LE encoding
|
|
Set-Content -Path "$FolderPath\desktop.ini" -Value $DesktopINI[$UserFolder] -Encoding Unicode -Force
|
|
(Get-Item -Path "$FolderPath\desktop.ini" -Force).Attributes = "Hidden", "System", "Archive"
|
|
(Get-Item -Path "$FolderPath\desktop.ini" -Force).Refresh()
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Show menu" function with the up/down arrow keys and enter key to make a selection
|
|
|
|
.EXAMPLE
|
|
ShowMenu -Menu $ListOfItems -Default $DefaultChoice
|
|
|
|
.LINK
|
|
https://qna.habr.com/answer?answer_id=1522379
|
|
#>
|
|
function ShowMenu
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter()]
|
|
[string]
|
|
$Title,
|
|
|
|
[Parameter(Mandatory = $true)]
|
|
[array]
|
|
$Menu,
|
|
|
|
[Parameter(Mandatory = $true)]
|
|
[int]
|
|
$Default
|
|
)
|
|
|
|
Write-Information -MessageData $Title -InformationAction Continue
|
|
|
|
$minY = [Console]::CursorTop
|
|
$y = [Math]::Max([Math]::Min($Default, $Menu.Count), 0)
|
|
do
|
|
{
|
|
[Console]::CursorTop = $minY
|
|
[Console]::CursorLeft = 0
|
|
$i = 0
|
|
foreach ($item in $Menu)
|
|
{
|
|
if ($i -ne $y)
|
|
{
|
|
Write-Information -MessageData (' {0}. {1} ' -f ($i+1), $item) -InformationAction Continue
|
|
}
|
|
else
|
|
{
|
|
Write-Information -MessageData ('[ {0}. {1} ]' -f ($i+1), $item) -InformationAction Continue
|
|
}
|
|
$i++
|
|
}
|
|
|
|
$k = [Console]::ReadKey()
|
|
switch ($k.Key)
|
|
{
|
|
"UpArrow"
|
|
{
|
|
if ($y -gt 0)
|
|
{
|
|
$y--
|
|
}
|
|
}
|
|
"DownArrow"
|
|
{
|
|
if ($y -lt ($Menu.Count - 1))
|
|
{
|
|
$y++
|
|
}
|
|
}
|
|
"Enter"
|
|
{
|
|
return $Menu[$y]
|
|
}
|
|
}
|
|
}
|
|
while ($k.Key -notin ([ConsoleKey]::Escape, [ConsoleKey]::Enter))
|
|
}
|
|
|
|
# Get the localized user folders names
|
|
$Signature = @{
|
|
Namespace = "WinAPI"
|
|
Name = "GetStr"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
[DllImport("kernel32.dll", CharSet = CharSet.Auto)]
|
|
public static extern IntPtr GetModuleHandle(string lpModuleName);
|
|
|
|
[DllImport("user32.dll", CharSet = CharSet.Auto)]
|
|
internal static extern int LoadString(IntPtr hInstance, uint uID, StringBuilder lpBuffer, int nBufferMax);
|
|
|
|
public static string GetString(uint strId)
|
|
{
|
|
IntPtr intPtr = GetModuleHandle("shell32.dll");
|
|
StringBuilder sb = new StringBuilder(255);
|
|
LoadString(intPtr, strId, sb, sb.Capacity);
|
|
return sb.ToString();
|
|
}
|
|
"@
|
|
}
|
|
if (-not ("WinAPI.GetStr" -as [type]))
|
|
{
|
|
Add-Type @Signature -Using System.Text
|
|
}
|
|
|
|
# The localized user folders names
|
|
$DesktopLocalizedString = [WinAPI.GetStr]::GetString(21769)
|
|
$DocumentsLocalizedString = [WinAPI.GetStr]::GetString(21770)
|
|
$DownloadsLocalizedString = [WinAPI.GetStr]::GetString(21798)
|
|
$MusicLocalizedString = [WinAPI.GetStr]::GetString(21790)
|
|
$PicturesLocalizedString = [WinAPI.GetStr]::GetString(21779)
|
|
$VideosLocalizedString = [WinAPI.GetStr]::GetString(21791)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Root"
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.RetrievingDrivesList -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
# Store all drives letters to use them within ShowMenu function
|
|
$DriveLetters = @((Get-Disk | Where-Object -FilterScript {$_.BusType -ne "USB"} | Get-Partition | Get-Volume | Where-Object -FilterScript {$null -ne $_.DriveLetter}).DriveLetter | Sort-Object)
|
|
|
|
# If the number of disks is more than one, set the second drive in the list as default drive
|
|
if ($DriveLetters.Count -gt 1)
|
|
{
|
|
$Script:Default = 1
|
|
}
|
|
else
|
|
{
|
|
$Script:Default = 0
|
|
}
|
|
|
|
# Desktop
|
|
Write-Verbose -Message ($Localization.DriveSelect -f $DesktopLocalizedString) -Verbose
|
|
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name Desktop
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $DesktopLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserFolderRequest -f $DesktopLocalizedString
|
|
$No = $Localization.No
|
|
$Yes = $Localization.Yes
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
$SelectedDrive = ShowMenu -Title ($Localization.DriveSelect -f $DesktopLocalizedString) -Menu $DriveLetters -Default $Script:Default
|
|
UserShellFolder -UserFolder Desktop -FolderPath "${SelectedDrive}:\Desktop" -RemoveDesktopINI
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Documents
|
|
Write-Verbose -Message ($Localization.DriveSelect -f $DocumentsLocalizedString) -Verbose
|
|
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name Personal
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $DocumentsLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserFolderRequest -f $DocumentsLocalizedString
|
|
$No = $Localization.No
|
|
$Yes = $Localization.Yes
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
$SelectedDrive = ShowMenu -Title ($Localization.DriveSelect -f $DocumentsLocalizedString) -Menu $DriveLetters -Default $Script:Default
|
|
UserShellFolder -UserFolder Documents -FolderPath "${SelectedDrive}:\Documents" -RemoveDesktopINI
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Downloads
|
|
Write-Verbose -Message ($Localization.DriveSelect -f $DownloadsLocalizedString) -Verbose
|
|
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "{374DE290-123F-4565-9164-39C4925E467B}"
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $DownloadsLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserFolderRequest -f $DownloadsLocalizedString
|
|
$No = $Localization.No
|
|
$Yes = $Localization.Yes
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
$SelectedDrive = ShowMenu -Title ($Localization.DriveSelect -f $DownloadsLocalizedString) -Menu $DriveLetters -Default $Script:Default
|
|
UserShellFolder -UserFolder Downloads -FolderPath "${SelectedDrive}:\Downloads" -RemoveDesktopINI
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Music
|
|
Write-Verbose -Message ($Localization.DriveSelect -f $MusicLocalizedString) -Verbose
|
|
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "My Music"
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $MusicLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserFolderRequest -f $MusicLocalizedString
|
|
$No = $Localization.No
|
|
$Yes = $Localization.Yes
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
$SelectedDrive = ShowMenu -Title ($Localization.DriveSelect -f $MusicLocalizedString) -Menu $DriveLetters -Default $Script:Default
|
|
UserShellFolder -UserFolder Music -FolderPath "${SelectedDrive}:\Music" -RemoveDesktopINI
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Pictures
|
|
Write-Verbose -Message ($Localization.DriveSelect -f $PicturesLocalizedString) -Verbose
|
|
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "My Pictures"
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $PicturesLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserFolderRequest -f $PicturesLocalizedString
|
|
$No = $Localization.No
|
|
$Yes = $Localization.Yes
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
$SelectedDrive = ShowMenu -Title ($Localization.DriveSelect -f $PicturesLocalizedString) -Menu $DriveLetters -Default $Script:Default
|
|
UserShellFolder -UserFolder Pictures -FolderPath "${SelectedDrive}:\Pictures" -RemoveDesktopINI
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Videos
|
|
Write-Verbose -Message ($Localization.DriveSelect -f $VideosLocalizedString) -Verbose
|
|
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "My Video"
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $VideosLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserFolderRequest -f $VideosLocalizedString
|
|
$No = $Localization.No
|
|
$Yes = $Localization.Yes
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
$SelectedDrive = ShowMenu -Title ($Localization.DriveSelect -f $VideosLocalizedString) -Menu $DriveLetters -Default $Script:Default
|
|
UserShellFolder -UserFolder Videos -FolderPath "${SelectedDrive}:\Videos" -RemoveDesktopINI
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
}
|
|
"Custom"
|
|
{
|
|
# Desktop
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name Desktop
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $DesktopLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserFolderSelect -f $DesktopLocalizedString
|
|
$Browse = $Localization.Browse
|
|
$No = $Localization.No
|
|
$Options = "&$Browse", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
$FolderBrowserDialog = New-Object -TypeName System.Windows.Forms.FolderBrowserDialog
|
|
$FolderBrowserDialog.Description = $Localization.FolderSelect
|
|
$FolderBrowserDialog.RootFolder = "MyComputer"
|
|
|
|
# Force move the open file dialog to the foreground
|
|
$Focus = New-Object -TypeName System.Windows.Forms.Form -Property @{TopMost = $true}
|
|
$FolderBrowserDialog.ShowDialog($Focus)
|
|
|
|
if ($FolderBrowserDialog.SelectedPath)
|
|
{
|
|
UserShellFolder -UserFolder Desktop -FolderPath $FolderBrowserDialog.SelectedPath -RemoveDesktopINI
|
|
Write-Verbose -Message ($Localization.NewUserFolderLocation -f $FolderBrowserDialog.SelectedPath) -Verbose
|
|
}
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Documents
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name Personal
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $DocumentsLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserFolderSelect -f $DocumentsLocalizedString
|
|
$Browse = $Localization.Browse
|
|
$No = $Localization.No
|
|
$Options = "&$Browse", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
$FolderBrowserDialog = New-Object -TypeName System.Windows.Forms.FolderBrowserDialog
|
|
$FolderBrowserDialog.Description = $Localization.FolderSelect
|
|
$FolderBrowserDialog.RootFolder = "MyComputer"
|
|
|
|
# Force move the open file dialog to the foreground
|
|
$Focus = New-Object -TypeName System.Windows.Forms.Form -Property @{TopMost = $true}
|
|
$FolderBrowserDialog.ShowDialog($Focus)
|
|
|
|
if ($FolderBrowserDialog.SelectedPath)
|
|
{
|
|
UserShellFolder -UserFolder Documents -FolderPath $FolderBrowserDialog.SelectedPath -RemoveDesktopINI
|
|
Write-Verbose -Message ($Localization.NewUserFolderLocation -f $FolderBrowserDialog.SelectedPath) -Verbose
|
|
}
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Downloads
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "{374DE290-123F-4565-9164-39C4925E467B}"
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $DownloadsLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserFolderSelect -f $DownloadsLocalizedString
|
|
$Browse = $Localization.Browse
|
|
$No = $Localization.No
|
|
$Options = "&$Browse", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
$FolderBrowserDialog = New-Object -TypeName System.Windows.Forms.FolderBrowserDialog
|
|
$FolderBrowserDialog.Description = $Localization.FolderSelect
|
|
$FolderBrowserDialog.RootFolder = "MyComputer"
|
|
|
|
# Force move the open file dialog to the foreground
|
|
$Focus = New-Object -TypeName System.Windows.Forms.Form -Property @{TopMost = $true}
|
|
$FolderBrowserDialog.ShowDialog($Focus)
|
|
|
|
if ($FolderBrowserDialog.SelectedPath)
|
|
{
|
|
UserShellFolder -UserFolder Downloads -FolderPath $FolderBrowserDialog.SelectedPath -RemoveDesktopINI
|
|
Write-Verbose -Message ($Localization.NewUserFolderLocation -f $FolderBrowserDialog.SelectedPath) -Verbose
|
|
}
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Music
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "My Music"
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $MusicLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserFolderSelect -f $MusicLocalizedString
|
|
$Browse = $Localization.Browse
|
|
$No = $Localization.No
|
|
$Options = "&$Browse", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
$FolderBrowserDialog = New-Object -TypeName System.Windows.Forms.FolderBrowserDialog
|
|
$FolderBrowserDialog.Description = $Localization.FolderSelect
|
|
$FolderBrowserDialog.RootFolder = "MyComputer"
|
|
|
|
# Force move the open file dialog to the foreground
|
|
$Focus = New-Object -TypeName System.Windows.Forms.Form -Property @{TopMost = $true}
|
|
$FolderBrowserDialog.ShowDialog($Focus)
|
|
|
|
if ($FolderBrowserDialog.SelectedPath)
|
|
{
|
|
UserShellFolder -UserFolder Music -FolderPath $FolderBrowserDialog.SelectedPath -RemoveDesktopINI
|
|
Write-Verbose -Message ($Localization.NewUserFolderLocation -f $FolderBrowserDialog.SelectedPath) -Verbose
|
|
}
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Pictures
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "My Pictures"
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $PicturesLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserFolderSelect -f $PicturesLocalizedString
|
|
$Browse = $Localization.Browse
|
|
$No = $Localization.No
|
|
$Options = "&$Browse", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
$FolderBrowserDialog = New-Object -TypeName System.Windows.Forms.FolderBrowserDialog
|
|
$FolderBrowserDialog.Description = $Localization.FolderSelect
|
|
$FolderBrowserDialog.RootFolder = "MyComputer"
|
|
|
|
# Force move the open file dialog to the foreground
|
|
$Focus = New-Object -TypeName System.Windows.Forms.Form -Property @{TopMost = $true}
|
|
$FolderBrowserDialog.ShowDialog($Focus)
|
|
|
|
if ($FolderBrowserDialog.SelectedPath)
|
|
{
|
|
UserShellFolder -UserFolder Pictures -FolderPath $FolderBrowserDialog.SelectedPath -RemoveDesktopINI
|
|
Write-Verbose -Message ($Localization.NewUserFolderLocation -f $FolderBrowserDialog.SelectedPath) -Verbose
|
|
}
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Videos
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "My Video"
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $VideosLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserFolderSelect -f $VideosLocalizedString
|
|
$Browse = $Localization.Browse
|
|
$No = $Localization.No
|
|
$Options = "&$Browse", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
$FolderBrowserDialog = New-Object -TypeName System.Windows.Forms.FolderBrowserDialog
|
|
$FolderBrowserDialog.Description = $Localization.FolderSelect
|
|
$FolderBrowserDialog.RootFolder = "MyComputer"
|
|
|
|
# Force move the open file dialog to the foreground
|
|
$Focus = New-Object -TypeName System.Windows.Forms.Form -Property @{TopMost = $true}
|
|
$FolderBrowserDialog.ShowDialog($Focus)
|
|
|
|
if ($FolderBrowserDialog.SelectedPath)
|
|
{
|
|
UserShellFolder -UserFolder Videos -FolderPath $FolderBrowserDialog.SelectedPath -RemoveDesktopINI
|
|
Write-Verbose -Message ($Localization.NewUserFolderLocation -f $FolderBrowserDialog.SelectedPath) -Verbose
|
|
}
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
}
|
|
"Default"
|
|
{
|
|
# Desktop
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name Desktop
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $DesktopLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserDefaultFolder -f $DesktopLocalizedString
|
|
$Yes = $Localization.Yes
|
|
$No = $Localization.No
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
UserShellFolder -UserFolder Desktop -FolderPath "$env:USERPROFILE\Desktop" -RemoveDesktopINI
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Documents
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name Personal
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $DocumentsLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserDefaultFolder -f $DocumentsLocalizedString
|
|
$Yes = $Localization.Yes
|
|
$No = $Localization.No
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
UserShellFolder -UserFolder Documents -FolderPath "$env:USERPROFILE\Documents" -RemoveDesktopINI
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Downloads
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "{374DE290-123F-4565-9164-39C4925E467B}"
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $DownloadsLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserDefaultFolder -f $DownloadsLocalizedString
|
|
$Yes = $Localization.Yes
|
|
$No = $Localization.No
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
UserShellFolder -UserFolder Downloads -FolderPath "$env:USERPROFILE\Downloads" -RemoveDesktopINI
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Music
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "My Music"
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $MusicLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserDefaultFolder -f $MusicLocalizedString
|
|
$Yes = $Localization.Yes
|
|
$No = $Localization.No
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
UserShellFolder -UserFolder Music -FolderPath "$env:USERPROFILE\Music" -RemoveDesktopINI
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Pictures
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "My Pictures"
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $PicturesLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserDefaultFolder -f $PicturesLocalizedString
|
|
$Yes = $Localization.Yes
|
|
$No = $Localization.No
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
UserShellFolder -UserFolder Pictures -FolderPath "$env:USERPROFILE\Pictures" -RemoveDesktopINI
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
|
|
# Videos
|
|
$CurrentUserFolderLocation = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "My Video"
|
|
Write-Verbose -Message ($Localization.CurrentUserFolderLocation -f $VideosLocalizedString, $CurrentUserFolderLocation) -Verbose
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.FilesWontBeMoved
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
$Title = ""
|
|
$Message = $Localization.UserDefaultFolder -f $VideosLocalizedString
|
|
$Yes = $Localization.Yes
|
|
$No = $Localization.No
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
UserShellFolder -UserFolder Videos -FolderPath "$env:USERPROFILE\Videos" -RemoveDesktopINI
|
|
}
|
|
"1"
|
|
{
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The location to save screenshots by pressing Win+PrtScr
|
|
|
|
.PARAMETER Desktop
|
|
Save screenshots by pressing Win+PrtScr on the Desktop
|
|
|
|
.PARAMETER Default
|
|
Save screenshots by pressing Win+PrtScr in the Pictures folder
|
|
|
|
.EXAMPLE
|
|
WinPrtScrFolder -Desktop
|
|
|
|
.EXAMPLE
|
|
WinPrtScrFolder -Default
|
|
|
|
.NOTES
|
|
The function will be applied only if the preset is configured to remove the OneDrive application, or the app was already uninstalled
|
|
otherwise the backup functionality for the "Desktop" and "Pictures" folders in OneDrive breaks
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function WinPrtScrFolder
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Desktop"
|
|
)]
|
|
[switch]
|
|
$Desktop,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Default"
|
|
)]
|
|
[switch]
|
|
$Default
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Desktop"
|
|
{
|
|
# Check how the script was invoked: via a preset or Function.ps1
|
|
$PresetName = (Get-PSCallStack).Position | Where-Object -FilterScript {
|
|
(($_.File -match ".ps1") -and ($_.File -notmatch "Functions.ps1")) -and (($_.Text -eq "WinPrtScrFolder -Desktop") -or ($_.Text -match "Invoke-Expression"))
|
|
}
|
|
if ($null -ne $PresetName)
|
|
{
|
|
# Get the name of a preset (e.g Sophia.ps1) regardless it was named
|
|
$PresetName = Split-Path -Path $PresetName.File -Leaf
|
|
# Check whether a preset contains the "OneDrive -Uninstall" string uncommented out
|
|
$OneDriveUninstallFunctionUncommented = (Get-Content -Path $PSScriptRoot\..\$PresetName -Encoding UTF8 -Force | Select-String -SimpleMatch "OneDrive -Uninstall").Line.StartsWith("#") -eq $false
|
|
$OneDriveInstalled = Get-Package -Name "Microsoft OneDrive" -ProviderName Programs -Force -ErrorAction Ignore
|
|
if ($OneDriveUninstallFunctionUncommented -or (-not $OneDriveInstalled))
|
|
{
|
|
$DesktopFolder = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name Desktop
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "{B7BEDE81-DF94-4682-A7D8-57A52620B86F}" -PropertyType ExpandString -Value $DesktopFolder -Force
|
|
}
|
|
else
|
|
{
|
|
Write-Warning -Message ($Localization.OneDriveWarning -f $MyInvocation.Line)
|
|
Write-Error -Message ($Localization.OneDriveWarning -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
else
|
|
{
|
|
# A preset file isn't taking a part so we ignore it and check only whether OneDrive was already uninstalled
|
|
if (-not (Get-Package -Name "Microsoft OneDrive" -ProviderName Programs -Force -ErrorAction Ignore))
|
|
{
|
|
$DesktopFolder = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name Desktop
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "{B7BEDE81-DF94-4682-A7D8-57A52620B86F}" -PropertyType ExpandString -Value $DesktopFolder -Force
|
|
}
|
|
else
|
|
{
|
|
Write-Warning -Message ($Localization.OneDriveWarning -f $MyInvocation.Line)
|
|
Write-Error -Message ($Localization.OneDriveWarning -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
}
|
|
"Default"
|
|
{
|
|
Remove-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "{B7BEDE81-DF94-4682-A7D8-57A52620B86F}" -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Recommended troubleshooter preferences
|
|
|
|
.PARAMETER Automatically
|
|
Run troubleshooter automatically, then notify me
|
|
|
|
.PARAMETER Default
|
|
Ask me before running troubleshooter
|
|
|
|
.EXAMPLE
|
|
RecommendedTroubleshooting -Automatically
|
|
|
|
.EXAMPLE
|
|
RecommendedTroubleshooting -Default
|
|
|
|
.NOTES
|
|
In order this feature to work the OS level of diagnostic data gathering will be set to "Optional diagnostic data" and the error reporting feature will be turned on
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function RecommendedTroubleshooting
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Automatically"
|
|
)]
|
|
[switch]
|
|
$Automatically,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Default"
|
|
)]
|
|
[switch]
|
|
$Default
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Automatically"
|
|
{
|
|
if (-not (Test-Path -Path HKLM:\SOFTWARE\Microsoft\WindowsMitigation))
|
|
{
|
|
New-Item -Path HKLM:\SOFTWARE\Microsoft\WindowsMitigation -Force
|
|
}
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\WindowsMitigation -Name UserPreference -PropertyType DWord -Value 3 -Force
|
|
}
|
|
"Default"
|
|
{
|
|
if (-not (Test-Path -Path HKLM:\SOFTWARE\Microsoft\WindowsMitigation))
|
|
{
|
|
New-Item -Path HKLM:\SOFTWARE\Microsoft\WindowsMitigation -Force
|
|
}
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\WindowsMitigation -Name UserPreference -PropertyType DWord -Value 2 -Force
|
|
}
|
|
}
|
|
|
|
# Set the OS level of diagnostic data gathering to "Optional diagnostic data"
|
|
New-ItemProperty -Path HKLM:\Software\Policies\Microsoft\Windows\DataCollection -Name AllowTelemetry -PropertyType DWord -Value 3 -Force
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\DataCollection -Name MaxTelemetryAllowed -PropertyType DWord -Value 3 -Force
|
|
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Diagnostics\DiagTrack -Name ShowedToastAtLevel -PropertyType DWord -Value 3 -Force
|
|
|
|
# Turn on Windows Error Reporting
|
|
Get-ScheduledTask -TaskName QueueReporting | Enable-ScheduledTask
|
|
Remove-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\Windows Error Reporting" -Name Disabled -Force -ErrorAction Ignore
|
|
|
|
Get-Service -Name WerSvc | Set-Service -StartupType Manual
|
|
Get-Service -Name WerSvc | Start-Service
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Folder windows launching in a separate process
|
|
|
|
.PARAMETER Enable
|
|
Launch folder windows in a separate process
|
|
|
|
.PARAMETER Disable
|
|
Do not launch folder windows in a separate process
|
|
|
|
.EXAMPLE
|
|
FoldersLaunchSeparateProcess -Enable
|
|
|
|
.EXAMPLE
|
|
FoldersLaunchSeparateProcess -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function FoldersLaunchSeparateProcess
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name SeparateProcess -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name SeparateProcess -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Reserved storage
|
|
|
|
.PARAMETER Disable
|
|
Disable and delete reserved storage after the next update installation
|
|
|
|
.PARAMETER Enable
|
|
Enable reserved storage after the next update installation
|
|
|
|
.EXAMPLE
|
|
ReservedStorage -Disable
|
|
|
|
.EXAMPLE
|
|
ReservedStorage -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function ReservedStorage
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
try
|
|
{
|
|
Set-WindowsReservedStorageState -State Disabled
|
|
}
|
|
catch [System.Runtime.InteropServices.COMException]
|
|
{
|
|
Write-Error -Message ($Localization.ReservedStorageIsInUse -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
"Enable"
|
|
{
|
|
Set-WindowsReservedStorageState -State Enabled
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Help look up via F1
|
|
|
|
.PARAMETER Disable
|
|
Disable help lookup via F1
|
|
|
|
.PARAMETER Enable
|
|
Enable help lookup via F1
|
|
|
|
.EXAMPLE
|
|
F1HelpPage -Disable
|
|
|
|
.EXAMPLE
|
|
F1HelpPage -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function F1HelpPage
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if (-not (Test-Path -Path "HKCU:\SOFTWARE\Classes\Typelib\{8cec5860-07a1-11d9-b15e-000d56bfe6ee}\1.0\0\win64"))
|
|
{
|
|
New-Item -Path "HKCU:\SOFTWARE\Classes\Typelib\{8cec5860-07a1-11d9-b15e-000d56bfe6ee}\1.0\0\win64" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Classes\Typelib\{8cec5860-07a1-11d9-b15e-000d56bfe6ee}\1.0\0\win64" -Name "(default)" -PropertyType String -Value "" -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
Remove-Item -Path "HKCU:\SOFTWARE\Classes\Typelib\{8cec5860-07a1-11d9-b15e-000d56bfe6ee}" -Recurse -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Num Lock at startup
|
|
|
|
.PARAMETER Enable
|
|
Enable Num Lock at startup
|
|
|
|
.PARAMETER Disable
|
|
Disable Num Lock at startup
|
|
|
|
.EXAMPLE
|
|
NumLock -Enable
|
|
|
|
.EXAMPLE
|
|
NumLock -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function NumLock
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path "Registry::HKEY_USERS\.DEFAULT\Control Panel\Keyboard" -Name InitialKeyboardIndicators -PropertyType String -Value 2147483650 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path "Registry::HKEY_USERS\.DEFAULT\Control Panel\Keyboard" -Name InitialKeyboardIndicators -PropertyType String -Value 2147483648 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Caps Lock
|
|
|
|
.PARAMETER Disable
|
|
Disable Caps Lock
|
|
|
|
.PARAMETER Enable
|
|
Enable Caps Lock
|
|
|
|
.EXAMPLE
|
|
CapsLock -Disable
|
|
|
|
.EXAMPLE
|
|
CapsLock -Enable
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function CapsLock
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path "HKLM:\SYSTEM\CurrentControlSet\Control\Keyboard Layout" -Name "Scancode Map" -PropertyType Binary -Value ([byte[]](0,0,0,0,0,0,0,0,2,0,0,0,0,0,58,0,0,0,0,0)) -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
Remove-ItemProperty -Path "HKLM:\SYSTEM\CurrentControlSet\Control\Keyboard Layout" -Name "Scancode Map" -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The shortcut to start Sticky Keys
|
|
|
|
.PARAMETER Disable
|
|
Turn off Sticky keys by pressing the Shift key 5 times
|
|
|
|
.PARAMETER Enable
|
|
Turn on Sticky keys by pressing the Shift key 5 times
|
|
|
|
.EXAMPLE
|
|
StickyShift -Disable
|
|
|
|
.EXAMPLE
|
|
StickyShift -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function StickyShift
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path "HKCU:\Control Panel\Accessibility\StickyKeys" -Name Flags -PropertyType String -Value 506 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path "HKCU:\Control Panel\Accessibility\StickyKeys" -Name Flags -PropertyType String -Value 510 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
AutoPlay for all media and devices
|
|
|
|
.PARAMETER Disable
|
|
Don't use AutoPlay for all media and devices
|
|
|
|
.PARAMETER Enable
|
|
Use AutoPlay for all media and devices
|
|
|
|
.EXAMPLE
|
|
Autoplay -Disable
|
|
|
|
.EXAMPLE
|
|
Autoplay -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function Autoplay
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\AutoplayHandlers -Name DisableAutoplay -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\AutoplayHandlers -Name DisableAutoplay -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Thumbnail cache removal
|
|
|
|
.PARAMETER Disable
|
|
Disable thumbnail cache removal
|
|
|
|
.PARAMETER Enable
|
|
Enable thumbnail cache removal
|
|
|
|
.EXAMPLE
|
|
ThumbnailCacheRemoval -Disable
|
|
|
|
.EXAMPLE
|
|
ThumbnailCacheRemoval -Enable
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function ThumbnailCacheRemoval
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\WOW6432Node\Microsoft\Windows\CurrentVersion\Explorer\VolumeCaches\Thumbnail Cache" -Name Autorun -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\WOW6432Node\Microsoft\Windows\CurrentVersion\Explorer\VolumeCaches\Thumbnail Cache" -Name Autorun -PropertyType DWord -Value 3 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Restart apps after signing in
|
|
|
|
.PARAMETER Enable
|
|
Automatically saving my restartable apps and restart them when I sign back in
|
|
|
|
.PARAMETER Disable
|
|
Turn off automatically saving my restartable apps and restart them when I sign back in
|
|
|
|
.EXAMPLE
|
|
SaveRestartableApps -Enable
|
|
|
|
.EXAMPLE
|
|
SaveRestartableApps -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function SaveRestartableApps
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon" -Name RestartApps -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows NT\CurrentVersion\Winlogon" -Name RestartApps -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Network Discovery File and Printers Sharing
|
|
|
|
.PARAMETER Enable
|
|
Enable "Network Discovery" and "File and Printers Sharing" for workgroup networks
|
|
|
|
.PARAMETER Disable
|
|
Disable "Network Discovery" and "File and Printers Sharing" for workgroup networks
|
|
|
|
.EXAMPLE
|
|
NetworkDiscovery -Enable
|
|
|
|
.EXAMPLE
|
|
NetworkDiscovery -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function NetworkDiscovery
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
$FirewallRules = @(
|
|
# File and printer sharing
|
|
"@FirewallAPI.dll,-32752",
|
|
|
|
# Network discovery
|
|
"@FirewallAPI.dll,-28502"
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
if (-not (Get-CimInstance -ClassName CIM_ComputerSystem).PartOfDomain)
|
|
{
|
|
Set-NetFirewallRule -Group $FirewallRules -Profile Private -Enabled True
|
|
|
|
Set-NetFirewallRule -Profile Public, Private -Name FPS-SMB-In-TCP -Enabled True
|
|
Set-NetConnectionProfile -NetworkCategory Private
|
|
}
|
|
}
|
|
"Disable"
|
|
{
|
|
if (-not (Get-CimInstance -ClassName CIM_ComputerSystem).PartOfDomain)
|
|
{
|
|
Set-NetFirewallRule -Group $FirewallRules -Profile Private -Enabled False
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Active hours
|
|
|
|
.PARAMETER Automatically
|
|
Automatically adjust active hours for me based on daily usage
|
|
|
|
.PARAMETER Manually
|
|
Manually adjust active hours for me based on daily usage
|
|
|
|
.EXAMPLE
|
|
ActiveHours -Automatically
|
|
|
|
.EXAMPLE
|
|
ActiveHours -Manually
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function ActiveHours
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Automatically"
|
|
)]
|
|
[switch]
|
|
$Automatically,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Manually"
|
|
)]
|
|
[switch]
|
|
$Manually
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Automatically"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\WindowsUpdate\UX\Settings -Name SmartActiveHoursState -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Manually"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\WindowsUpdate\UX\Settings -Name SmartActiveHoursState -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Restart as soon as possible to finish updating
|
|
|
|
.PARAMETER Enable
|
|
Restart as soon as possible to finish updating
|
|
|
|
.PARAMETER Disable
|
|
Don't restart as soon as possible to finish updating
|
|
|
|
.EXAMPLE
|
|
DeviceRestartAfterUpdate -Enable
|
|
|
|
.EXAMPLE
|
|
DeviceRestartAfterUpdate -Disable
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function RestartDeviceAfterUpdate
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\WindowsUpdate\UX\Settings -Name IsExpedited -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\WindowsUpdate\UX\Settings -Name IsExpedited -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Register app, calculate hash, and associate with an extension with the "How do you want to open this" pop-up hidden
|
|
|
|
.PARAMETER ProgramPath
|
|
Set a path to a program to associate an extension with
|
|
|
|
.PARAMETER Extension
|
|
Set the extension type
|
|
|
|
.PARAMETER Icon
|
|
Set a path to an icon
|
|
|
|
.EXAMPLE
|
|
Set-Association -ProgramPath "C:\SumatraPDF.exe" -Extension .pdf -Icon "shell32.dll,100"
|
|
|
|
.EXAMPLE
|
|
Set-Association -ProgramPath "%ProgramFiles%\Notepad++\notepad++.exe" -Extension .txt -Icon "%ProgramFiles%\Notepad++\notepad++.exe,0"
|
|
|
|
.LINK
|
|
https://github.com/DanysysTeam/PS-SFTA
|
|
https://github.com/default-username-was-already-taken/set-fileassoc
|
|
https://forum.ru-board.com/profile.cgi?action=show&member=westlife
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function Set-Association
|
|
{
|
|
[CmdletBinding()]
|
|
Param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 0
|
|
)]
|
|
[String]
|
|
$ProgramPath,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 1
|
|
)]
|
|
[String]
|
|
$Extension,
|
|
|
|
[Parameter(Mandatory = $false)]
|
|
[String]
|
|
$Icon
|
|
)
|
|
|
|
$ProgramPath = [System.Environment]::ExpandEnvironmentVariables($ProgramPath)
|
|
$Icon = [System.Environment]::ExpandEnvironmentVariables($Icon)
|
|
|
|
if (Test-Path -Path $ProgramPath)
|
|
{
|
|
# Generate ProgId
|
|
$ProgId = (Get-Item -Path $ProgramPath).BaseName + $Extension.ToUpper()
|
|
}
|
|
else
|
|
{
|
|
$ProgId = $ProgramPath
|
|
}
|
|
|
|
#region functions
|
|
$RegistryUtils = @'
|
|
using System;
|
|
using System.Runtime.InteropServices;
|
|
using System.Security.AccessControl;
|
|
using System.Text;
|
|
using Microsoft.Win32;
|
|
using FILETIME = System.Runtime.InteropServices.ComTypes.FILETIME;
|
|
|
|
namespace RegistryUtils
|
|
{
|
|
public static class Action
|
|
{
|
|
[DllImport("advapi32.dll", CharSet = CharSet.Auto)]
|
|
private static extern int RegOpenKeyEx(UIntPtr hKey, string subKey, int ulOptions, int samDesired, out UIntPtr hkResult);
|
|
|
|
[DllImport("advapi32.dll", EntryPoint = "RegQueryInfoKey", CallingConvention = CallingConvention.Winapi, SetLastError = true)]
|
|
private static extern int RegQueryInfoKey(UIntPtr hkey, out StringBuilder lpClass, ref uint lpcbClass, IntPtr lpReserved,
|
|
out uint lpcSubKeys, out uint lpcbMaxSubKeyLen, out uint lpcbMaxClassLen, out uint lpcValues, out uint lpcbMaxValueNameLen,
|
|
out uint lpcbMaxValueLen, out uint lpcbSecurityDescriptor, ref FILETIME lpftLastWriteTime);
|
|
|
|
[DllImport("advapi32.dll", SetLastError = true)]
|
|
private static extern int RegCloseKey(UIntPtr hKey);
|
|
|
|
[DllImport("advapi32.dll", SetLastError=true, CharSet = CharSet.Unicode)]
|
|
private static extern uint RegDeleteKey(UIntPtr hKey, string subKey);
|
|
|
|
public static void DeleteKey(RegistryHive registryHive, string subkey) {
|
|
UIntPtr hKey = UIntPtr.Zero;
|
|
var hive = new UIntPtr(unchecked((uint)registryHive));
|
|
RegOpenKeyEx(hive, subkey, 0, 0x20019, out hKey);
|
|
RegDeleteKey(hive, subkey);
|
|
}
|
|
|
|
private static DateTime ToDateTime(FILETIME ft)
|
|
{
|
|
IntPtr buf = IntPtr.Zero;
|
|
try
|
|
{
|
|
long[] longArray = new long[1];
|
|
int cb = Marshal.SizeOf(ft);
|
|
buf = Marshal.AllocHGlobal(cb);
|
|
Marshal.StructureToPtr(ft, buf, false);
|
|
Marshal.Copy(buf, longArray, 0, 1);
|
|
return DateTime.FromFileTime(longArray[0]);
|
|
}
|
|
finally
|
|
{
|
|
if (buf != IntPtr.Zero) Marshal.FreeHGlobal(buf);
|
|
}
|
|
}
|
|
|
|
public static DateTime? GetLastModified(RegistryHive registryHive, string subKey)
|
|
{
|
|
var lastModified = new FILETIME();
|
|
var lpcbClass = new uint();
|
|
var lpReserved = new IntPtr();
|
|
UIntPtr key = UIntPtr.Zero;
|
|
|
|
try
|
|
{
|
|
try
|
|
{
|
|
var hive = new UIntPtr(unchecked((uint)registryHive));
|
|
if (RegOpenKeyEx(hive, subKey, 0, (int)RegistryRights.ReadKey, out key) != 0)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
uint lpcbSubKeys;
|
|
uint lpcbMaxKeyLen;
|
|
uint lpcbMaxClassLen;
|
|
uint lpcValues;
|
|
uint maxValueName;
|
|
uint maxValueLen;
|
|
uint securityDescriptor;
|
|
StringBuilder sb;
|
|
|
|
if (RegQueryInfoKey(key, out sb, ref lpcbClass, lpReserved, out lpcbSubKeys, out lpcbMaxKeyLen, out lpcbMaxClassLen, out lpcValues, out maxValueName, out maxValueLen, out securityDescriptor, ref lastModified) != 0)
|
|
{
|
|
return null;
|
|
}
|
|
|
|
var result = ToDateTime(lastModified);
|
|
return result;
|
|
}
|
|
finally
|
|
{
|
|
if (key != UIntPtr.Zero)
|
|
{
|
|
RegCloseKey(key);
|
|
}
|
|
}
|
|
}
|
|
catch (Exception)
|
|
{
|
|
return null;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
'@
|
|
|
|
if (-not('RegistryUtils.Action' -as [type]))
|
|
{
|
|
Add-Type -TypeDefinition $RegistryUtils
|
|
}
|
|
|
|
function Set-Icon
|
|
{
|
|
Param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 0
|
|
|
|
)]
|
|
[String]
|
|
$ProgId,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 1
|
|
)]
|
|
[String]
|
|
$Icon
|
|
)
|
|
|
|
if (-not (Test-Path -Path "HKCU:\SOFTWARE\Classes\$ProgId\DefaultIcon"))
|
|
{
|
|
New-Item -Path "HKCU:\SOFTWARE\Classes\$ProgId\DefaultIcon" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Classes\$ProgId\DefaultIcon" -Name "(default)" -PropertyType String -Value $Icon -Force
|
|
}
|
|
|
|
function Remove-UserChoiceKey
|
|
{
|
|
Param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 0
|
|
)]
|
|
[string]
|
|
$SubKey
|
|
)
|
|
|
|
[RegistryUtils.Action]::DeleteKey([Microsoft.Win32.RegistryHive]::CurrentUser,$SubKey)
|
|
}
|
|
|
|
function Set-UserAccessKey
|
|
{
|
|
Param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 0
|
|
)]
|
|
[string]
|
|
$SubKey
|
|
)
|
|
|
|
$OpenSubKey = [Microsoft.Win32.Registry]::CurrentUser.OpenSubKey($SubKey,'ReadWriteSubTree','TakeOwnership')
|
|
if ($OpenSubKey)
|
|
{
|
|
$Acl = [System.Security.AccessControl.RegistrySecurity]::new()
|
|
# Get current user SID
|
|
$UserSID = (Get-CimInstance -ClassName Win32_UserAccount | Where-Object -FilterScript {$_.Name -eq $env:USERNAME}).SID
|
|
$Acl.SetSecurityDescriptorSddlForm("O:$UserSID`G:$UserSID`D:AI(D;;DC;;;$UserSID)")
|
|
$OpenSubKey.SetAccessControl($Acl)
|
|
$OpenSubKey.Close()
|
|
}
|
|
}
|
|
|
|
function Write-ExtensionKeys
|
|
{
|
|
Param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 0
|
|
)]
|
|
[string]
|
|
$ProgId,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 1
|
|
)]
|
|
[String]
|
|
$Extension
|
|
)
|
|
|
|
$OrigProgID = (Get-ItemProperty -Path "HKLM:\SOFTWARE\Classes\$Extension" -Name "(default)" -ErrorAction Ignore)."(default)"
|
|
if ($OrigProgID)
|
|
{
|
|
# Save possible ProgIds history with extension
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ApplicationAssociationToasts" -Name "$ProgID_$Extension" -PropertyType DWord -Value 0 -Force
|
|
}
|
|
|
|
$Name = "{0}_$Extension" -f (Split-Path -Path $ProgId -Leaf)
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ApplicationAssociationToasts" -Name $Name -PropertyType DWord -Value 0 -Force
|
|
|
|
if ("$ProgId_$Extension" -ne $Name)
|
|
{
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ApplicationAssociationToasts" -Name "$ProgId_$Extension" -PropertyType DWord -Value 0 -Force
|
|
}
|
|
|
|
# If ProgId doesn't exist set the specified ProgId for the extensions
|
|
if (-not $OrigProgID)
|
|
{
|
|
if (-not (Test-Path -Path "HKCU:\SOFTWARE\Classes\$Extension"))
|
|
{
|
|
New-Item -Path "HKCU:\SOFTWARE\Classes\$Extension" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Classes\$Extension" -Name "(default)" -PropertyType String -Value $ProgId -Force
|
|
}
|
|
|
|
# Set the specified ProgId in the possible options for the assignment
|
|
if (-not (Test-Path -Path "HKCU:\SOFTWARE\Classes\$Extension\OpenWithProgids"))
|
|
{
|
|
New-Item -Path "HKCU:\SOFTWARE\Classes\$Extension\OpenWithProgids" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Classes\$Extension\OpenWithProgids" -Name $ProgId -PropertyType None -Value ([byte[]]@()) -Force
|
|
|
|
# Set the system ProgId to the extension parameters for the File Explorer to the possible options for the assignment, and if absent set the specified ProgId
|
|
if ($OrigProgID)
|
|
{
|
|
if (-not (Test-Path -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\OpenWithProgids"))
|
|
{
|
|
New-Item -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\OpenWithProgids" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\OpenWithProgids" -Name $OrigProgID -PropertyType None -Value ([byte[]]@()) -Force
|
|
}
|
|
|
|
if (-not (Test-Path -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\OpenWithProgids"))
|
|
{
|
|
New-Item -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\OpenWithProgids" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\OpenWithProgids" -Name $ProgID -PropertyType None -Value ([byte[]]@()) -Force
|
|
|
|
# Removing the UserChoice key
|
|
Remove-UserChoiceKey -SubKey "Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\UserChoice"
|
|
|
|
# Setting parameters in UserChoice. The key is being autocreated
|
|
if (-not (Test-Path -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\UserChoice"))
|
|
{
|
|
New-Item -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\UserChoice" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\UserChoice" -Name ProgId -PropertyType String -Value $ProgID -Force
|
|
|
|
# Getting a hash based on the time of the section's last modification. After creating and setting the first parameter
|
|
$ProgHash = Get-Hash -ProgId $ProgId -Extension $Extension -SubKey "Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\UserChoice"
|
|
|
|
if (-not (Test-Path -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\UserChoice"))
|
|
{
|
|
New-Item -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\UserChoice" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\UserChoice" -Name Hash -PropertyType String -Value $ProgHash -Force
|
|
|
|
# Setting a ban on changing the UserChoice section
|
|
Set-UserAccessKey -SubKey "Software\Microsoft\Windows\CurrentVersion\Explorer\FileExts\$Extension\UserChoice"
|
|
}
|
|
|
|
function Write-AdditionalKeys
|
|
{
|
|
Param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 0
|
|
)]
|
|
[string]
|
|
$ProgId,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 1
|
|
)]
|
|
[string]
|
|
$Extension
|
|
)
|
|
|
|
# If there is the system extension ProgId, write it to the already configured by default
|
|
if ((Get-ItemProperty -Path "HKLM:\SOFTWARE\Classes\$Extension" -Name "(default)" -ErrorAction Ignore)."(default)")
|
|
{
|
|
if (-not (Test-Path -Path Registry::HKEY_USERS\.DEFAULT\Software\Microsoft\Windows\CurrentVersion\FileAssociations\ProgIds))
|
|
{
|
|
New-Item -Path Registry::HKEY_USERS\.DEFAULT\Software\Microsoft\Windows\CurrentVersion\FileAssociations\ProgIds -Force
|
|
}
|
|
New-ItemProperty -Path Registry::HKEY_USERS\.DEFAULT\Software\Microsoft\Windows\CurrentVersion\FileAssociations\ProgIds -Name "_$Extension" -PropertyType DWord -Value 1 -Force
|
|
}
|
|
|
|
# Setting 'NoOpenWith' for all registered the extension ProgIDs
|
|
[psobject]$OpenSubkey = Get-Item -Path "Registry::HKEY_CLASSES_ROOT\$Extension\OpenWithProgids" -ErrorAction Ignore | Select-Object -ExpandProperty Property
|
|
|
|
if ($OpenSubkey)
|
|
{
|
|
foreach ($AppxProgID in ($OpenSubkey | Where-Object -FilterScript {$_ -match "AppX"}))
|
|
{
|
|
# If an app is installed
|
|
if (Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Classes\$AppxProgID\Shell\open" -Name PackageId)
|
|
{
|
|
# If the specified ProgId is equal to UWP installed ProgId
|
|
if ($ProgId -eq $AppxProgID)
|
|
{
|
|
# Remove association limitations for this UWP apps
|
|
Remove-ItemProperty -Path "HKCU:\SOFTWARE\Classes\$AppxProgID" -Name NoOpenWith -Force -ErrorAction Ignore
|
|
Remove-ItemProperty -Path "HKCU:\SOFTWARE\Classes\$AppxProgID" -Name NoStaticDefaultVerb -Force -ErrorAction Ignore
|
|
}
|
|
else
|
|
{
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Classes\$AppxProgID" -Name NoOpenWith -PropertyType String -Value "" -Force
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
$picture = (Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\KindMap" -Name $Extension -ErrorAction Ignore).$Extension
|
|
$PBrush = (Get-ItemProperty -Path "HKLM:\SOFTWARE\Classes\PBrush\CLSID" -Name "(default)" -ErrorAction Ignore)."(default)"
|
|
|
|
if (($picture -eq "picture") -and $PBrush)
|
|
{
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ApplicationAssociationToasts" -Name "PBrush_$Extension" -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
|
|
function Get-Hash
|
|
{
|
|
[CmdletBinding()]
|
|
[OutputType([string])]
|
|
Param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 0
|
|
)]
|
|
[string]
|
|
$ProgId,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 1
|
|
)]
|
|
[string] $Extension,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
Position = 2
|
|
)]
|
|
[string]
|
|
$SubKey
|
|
)
|
|
|
|
$PatentHash = @'
|
|
using System;
|
|
|
|
namespace FileAssoc
|
|
{
|
|
public static class PatentHash
|
|
{
|
|
public static uint[] WordSwap(byte[] a, int sz, byte[] md5)
|
|
{
|
|
if (sz < 2 || (sz & 1) == 1) {
|
|
throw new ArgumentException(String.Format("Invalid input size: {0}", sz), "sz");
|
|
}
|
|
|
|
unchecked {
|
|
uint o1 = 0;
|
|
uint o2 = 0;
|
|
int ta = 0;
|
|
int ts = sz;
|
|
int ti = ((sz - 2) >> 1) + 1;
|
|
|
|
uint c0 = (BitConverter.ToUInt32(md5, 0) | 1) + 0x69FB0000;
|
|
uint c1 = (BitConverter.ToUInt32(md5, 4) | 1) + 0x13DB0000;
|
|
|
|
for (uint i = (uint)ti; i > 0; i--) {
|
|
uint n = BitConverter.ToUInt32(a, ta) + o1;
|
|
ta += 8;
|
|
ts -= 2;
|
|
|
|
uint v1 = 0x79F8A395 * (n * c0 - 0x10FA9605 * (n >> 16)) + 0x689B6B9F * ((n * c0 - 0x10FA9605 * (n >> 16)) >> 16);
|
|
uint v2 = 0xEA970001 * v1 - 0x3C101569 * (v1 >> 16);
|
|
uint v3 = BitConverter.ToUInt32(a, ta - 4) + v2;
|
|
uint v4 = v3 * c1 - 0x3CE8EC25 * (v3 >> 16);
|
|
uint v5 = 0x59C3AF2D * v4 - 0x2232E0F1 * (v4 >> 16);
|
|
|
|
o1 = 0x1EC90001 * v5 + 0x35BD1EC9 * (v5 >> 16);
|
|
o2 += o1 + v2;
|
|
}
|
|
|
|
if (ts == 1) {
|
|
uint n = BitConverter.ToUInt32(a, ta) + o1;
|
|
|
|
uint v1 = n * c0 - 0x10FA9605 * (n >> 16);
|
|
uint v2 = 0xEA970001 * (0x79F8A395 * v1 + 0x689B6B9F * (v1 >> 16)) - 0x3C101569 * ((0x79F8A395 * v1 + 0x689B6B9F * (v1 >> 16)) >> 16);
|
|
uint v3 = v2 * c1 - 0x3CE8EC25 * (v2 >> 16);
|
|
|
|
o1 = 0x1EC90001 * (0x59C3AF2D * v3 - 0x2232E0F1 * (v3 >> 16)) + 0x35BD1EC9 * ((0x59C3AF2D * v3 - 0x2232E0F1 * (v3 >> 16)) >> 16);
|
|
o2 += o1 + v2;
|
|
}
|
|
|
|
uint[] ret = new uint[2];
|
|
ret[0] = o1;
|
|
ret[1] = o2;
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
public static uint[] Reversible(byte[] a, int sz, byte[] md5)
|
|
{
|
|
if (sz < 2 || (sz & 1) == 1) {
|
|
throw new ArgumentException(String.Format("Invalid input size: {0}", sz), "sz");
|
|
}
|
|
|
|
unchecked {
|
|
uint o1 = 0;
|
|
uint o2 = 0;
|
|
int ta = 0;
|
|
int ts = sz;
|
|
int ti = ((sz - 2) >> 1) + 1;
|
|
|
|
uint c0 = BitConverter.ToUInt32(md5, 0) | 1;
|
|
uint c1 = BitConverter.ToUInt32(md5, 4) | 1;
|
|
|
|
for (uint i = (uint)ti; i > 0; i--) {
|
|
uint n = (BitConverter.ToUInt32(a, ta) + o1) * c0;
|
|
n = 0xB1110000 * n - 0x30674EEF * (n >> 16);
|
|
ta += 8;
|
|
ts -= 2;
|
|
|
|
uint v1 = 0x5B9F0000 * n - 0x78F7A461 * (n >> 16);
|
|
uint v2 = 0x1D830000 * (0x12CEB96D * (v1 >> 16) - 0x46930000 * v1) + 0x257E1D83 * ((0x12CEB96D * (v1 >> 16) - 0x46930000 * v1) >> 16);
|
|
uint v3 = BitConverter.ToUInt32(a, ta - 4) + v2;
|
|
|
|
uint v4 = 0x16F50000 * c1 * v3 - 0x5D8BE90B * (c1 * v3 >> 16);
|
|
uint v5 = 0x2B890000 * (0x96FF0000 * v4 - 0x2C7C6901 * (v4 >> 16)) + 0x7C932B89 * ((0x96FF0000 * v4 - 0x2C7C6901 * (v4 >> 16)) >> 16);
|
|
|
|
o1 = 0x9F690000 * v5 - 0x405B6097 * (v5 >> 16);
|
|
o2 += o1 + v2;
|
|
}
|
|
|
|
if (ts == 1) {
|
|
uint n = BitConverter.ToUInt32(a, ta) + o1;
|
|
|
|
uint v1 = 0xB1110000 * c0 * n - 0x30674EEF * ((c0 * n) >> 16);
|
|
uint v2 = 0x5B9F0000 * v1 - 0x78F7A461 * (v1 >> 16);
|
|
uint v3 = 0x1D830000 * (0x12CEB96D * (v2 >> 16) - 0x46930000 * v2) + 0x257E1D83 * ((0x12CEB96D * (v2 >> 16) - 0x46930000 * v2) >> 16);
|
|
uint v4 = 0x16F50000 * c1 * v3 - 0x5D8BE90B * ((c1 * v3) >> 16);
|
|
uint v5 = 0x96FF0000 * v4 - 0x2C7C6901 * (v4 >> 16);
|
|
|
|
o1 = 0x9F690000 * (0x2B890000 * v5 + 0x7C932B89 * (v5 >> 16)) - 0x405B6097 * ((0x2B890000 * v5 + 0x7C932B89 * (v5 >> 16)) >> 16);
|
|
o2 += o1 + v2;
|
|
}
|
|
|
|
uint[] ret = new uint[2];
|
|
ret[0] = o1;
|
|
ret[1] = o2;
|
|
return ret;
|
|
}
|
|
}
|
|
|
|
public static long MakeLong(uint left, uint right) {
|
|
return (long)left << 32 | (long)right;
|
|
}
|
|
}
|
|
}
|
|
'@
|
|
|
|
if ( -not ('FileAssoc.PatentHash' -as [type]))
|
|
{
|
|
Add-Type -TypeDefinition $PatentHash
|
|
}
|
|
|
|
function Get-KeyLastWriteTime ($SubKey)
|
|
{
|
|
$LM = [RegistryUtils.Action]::GetLastModified([Microsoft.Win32.RegistryHive]::CurrentUser,$SubKey)
|
|
$FT = ([DateTime]::New($LM.Year, $LM.Month, $LM.Day, $LM.Hour, $LM.Minute, 0, $LM.Kind)).ToFileTime()
|
|
|
|
return [string]::Format("{0:x8}{1:x8}", $FT -shr 32, $FT -band [uint32]::MaxValue)
|
|
}
|
|
|
|
function Get-DataArray
|
|
{
|
|
[OutputType([array])]
|
|
|
|
# Secret static string stored in %SystemRoot%\SysWOW64\shell32.dll
|
|
$userExperience = "User Choice set via Windows User Experience {D18B6DD5-6124-4341-9318-804003BAFA0B}"
|
|
# Get user SID
|
|
$userSid = (Get-CimInstance -ClassName Win32_UserAccount | Where-Object -FilterScript {$_.Name -eq $env:USERNAME}).SID
|
|
$KeyLastWriteTime = Get-KeyLastWriteTime -SubKey $SubKey
|
|
$baseInfo = ("{0}{1}{2}{3}{4}" -f $Extension, $userSid, $ProgId, $KeyLastWriteTime, $userExperience).ToLowerInvariant()
|
|
$StringToUTF16LEArray = [System.Collections.ArrayList]@([System.Text.Encoding]::Unicode.GetBytes($baseInfo))
|
|
$StringToUTF16LEArray += (0,0)
|
|
|
|
return $StringToUTF16LEArray
|
|
}
|
|
|
|
function Get-PatentHash
|
|
{
|
|
[OutputType([string])]
|
|
param
|
|
(
|
|
[Parameter(Mandatory = $true)]
|
|
[byte[]]
|
|
$A,
|
|
|
|
[Parameter(Mandatory = $true)]
|
|
[byte[]]
|
|
$MD5
|
|
)
|
|
|
|
$Size = $A.Count
|
|
$ShiftedSize = ($Size -shr 2) - ($Size -shr 2 -band 1) * 1
|
|
|
|
[uint32[]]$A1 = [FileAssoc.PatentHash]::WordSwap($A, [int]$ShiftedSize, $MD5)
|
|
[uint32[]]$A2 = [FileAssoc.PatentHash]::Reversible($A, [int]$ShiftedSize, $MD5)
|
|
|
|
$Ret = [FileAssoc.PatentHash]::MakeLong($A1[1] -bxor $A2[1], $A1[0] -bxor $A2[0])
|
|
|
|
return [System.Convert]::ToBase64String([System.BitConverter]::GetBytes([Int64]$Ret))
|
|
}
|
|
|
|
$DataArray = Get-DataArray
|
|
$DataMD5 = [System.Security.Cryptography.HashAlgorithm]::Create("MD5").ComputeHash($DataArray)
|
|
$Hash = Get-PatentHash -A $DataArray -MD5 $DataMD5
|
|
|
|
return $Hash
|
|
}
|
|
#endregion functions
|
|
|
|
if ($ProgramPath)
|
|
{
|
|
if (-not (Test-Path -Path "HKCU:\SOFTWARE\Classes\$ProgId\shell\open\command"))
|
|
{
|
|
New-Item -Path "HKCU:\SOFTWARE\Classes\$ProgId\shell\open\command" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Classes\$ProgId\shell\open\command" -Name "(Default)" -PropertyType String -Value "`"$ProgramPath`" `"%1`"" -Force
|
|
|
|
$FileNameEXE = Split-Path -Path $ProgramPath -Leaf
|
|
if (-not (Test-Path -Path "HKCU:\SOFTWARE\Classes\Applications\$FileNameEXE\shell\open\command"))
|
|
{
|
|
New-Item -Path "HKCU:\SOFTWARE\Classes\Applications\$FileNameEXE\shell\open\command" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Classes\Applications\$FileNameEXE\shell\open\command" -Name "(Default)" -PropertyType String -Value "`"$ProgramPath`" `"%1`"" -Force
|
|
}
|
|
|
|
if ($Icon)
|
|
{
|
|
Set-Icon -ProgId $ProgId -Icon $Icon
|
|
}
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
# Setting additional parameters to comply with the requirements before configuring the extension
|
|
Write-AdditionalKeys -ProgId $ProgId -Extension $Extension
|
|
|
|
# If the file extension specified configure the extension
|
|
Write-ExtensionKeys -ProgId $ProgId -Extension $Extension
|
|
|
|
# Refresh the desktop icons
|
|
$UpdateExplorer = @{
|
|
Namespace = "WinAPI"
|
|
Name = "UpdateExplorer"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
[DllImport("shell32.dll", CharSet = CharSet.Auto, SetLastError = false)]
|
|
private static extern int SHChangeNotify(int eventId, int flags, IntPtr item1, IntPtr item2);
|
|
|
|
public static void Refresh()
|
|
{
|
|
// Update desktop icons
|
|
SHChangeNotify(0x8000000, 0x1000, IntPtr.Zero, IntPtr.Zero);
|
|
}
|
|
"@
|
|
}
|
|
if (-not ("WinAPI.UpdateExplorer" -as [type]))
|
|
{
|
|
Add-Type @UpdateExplorer
|
|
}
|
|
|
|
[WinAPI.UpdateExplorer]::Refresh()
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Default terminal app
|
|
|
|
.PARAMETER WindowsTerminal
|
|
Set Windows Terminal as default terminal app to host the user interface for command-line applications
|
|
|
|
.PARAMETER ConsoleHost
|
|
Set Windows Console Host as default terminal app to host the user interface for command-line applications
|
|
|
|
.EXAMPLE
|
|
DefaultTerminalApp -WindowsTerminal
|
|
|
|
.EXAMPLE
|
|
DefaultTerminalApp -ConsoleHost
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function DefaultTerminalApp
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "WindowsTerminal"
|
|
)]
|
|
[switch]
|
|
$WindowsTerminal,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "ConsoleHost"
|
|
)]
|
|
[switch]
|
|
$ConsoleHost
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"WindowsTerminal"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.WindowsTerminal)
|
|
{
|
|
# Checking if the Terminal version supports such feature
|
|
$TerminalVersion = (Get-AppxPackage -Name Microsoft.WindowsTerminal).Version
|
|
if ([System.Version]$TerminalVersion -ge [System.Version]"1.11")
|
|
{
|
|
if (-not (Test-Path -Path "HKCU:\Console\%%Startup"))
|
|
{
|
|
New-Item -Path "HKCU:\Console\%%Startup" -Force
|
|
}
|
|
|
|
# Find the current GUID of Windows Terminal
|
|
$PackageFullName = (Get-AppxPackage -Name Microsoft.WindowsTerminal).PackageFullName
|
|
Get-ChildItem -Path "HKLM:\SOFTWARE\Classes\PackagedCom\Package\$PackageFullName\Class" | ForEach-Object -Process {
|
|
if ((Get-ItemPropertyValue -Path $_.PSPath -Name ServerId) -eq 0)
|
|
{
|
|
New-ItemProperty -Path "HKCU:\Console\%%Startup" -Name DelegationConsole -PropertyType String -Value $_.PSChildName -Force
|
|
}
|
|
|
|
if ((Get-ItemPropertyValue -Path $_.PSPath -Name ServerId) -eq 1)
|
|
{
|
|
New-ItemProperty -Path "HKCU:\Console\%%Startup" -Name DelegationTerminal -PropertyType String -Value $_.PSChildName -Force
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
"ConsoleHost"
|
|
{
|
|
New-ItemProperty -Path "HKCU:\Console\%%Startup" -Name DelegationConsole -PropertyType String -Value "{00000000-0000-0000-0000-000000000000}" -Force
|
|
New-ItemProperty -Path "HKCU:\Console\%%Startup" -Name DelegationTerminal -PropertyType String -Value "{00000000-0000-0000-0000-000000000000}" -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Install the latest Microsoft Visual C++ Redistributable Packages 2015–2022 x64
|
|
|
|
.EXAMPLE
|
|
InstallVCRedistx64
|
|
|
|
.LINK
|
|
https://docs.microsoft.com/en-us/cpp/windows/latest-supported-vc-redist
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function InstallVCRedistx64
|
|
{
|
|
try
|
|
{
|
|
# Check the internet connection
|
|
$Parameters = @{
|
|
Uri = "https://www.google.com"
|
|
Method = "Head"
|
|
DisableKeepAlive = $true
|
|
UseBasicParsing = $true
|
|
}
|
|
if (-not (Invoke-WebRequest @Parameters).StatusDescription)
|
|
{
|
|
return
|
|
}
|
|
|
|
if ([System.Version](Get-AppxPackage -Name Microsoft.DesktopAppInstaller).Version -ge [System.Version]"1.17")
|
|
{
|
|
winget install --id=Microsoft.VC++2015-2022Redist-x64 --exact --accept-source-agreements
|
|
}
|
|
else
|
|
{
|
|
$DownloadsFolder = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "{374DE290-123F-4565-9164-39C4925E467B}"
|
|
$Parameters = @{
|
|
Uri = "https://aka.ms/vs/17/release/VC_redist.x64.exe"
|
|
OutFile = "$DownloadsFolder\VC_redist.x64.exe"
|
|
UseBasicParsing = $true
|
|
Verbose = $true
|
|
}
|
|
Invoke-WebRequest @Parameters
|
|
|
|
Start-Process -FilePath "$DownloadsFolder\VC_redist.x64.exe" -ArgumentList "/install /passive /norestart" -Wait
|
|
|
|
<#
|
|
PowerShell 5.1 (7.2 too) interprets the 8.3 file name literally, if an environment variable contains a non-latin word,
|
|
so you won't be able to remove "$env:TEMP\dd_vcredist_amd64_*.log" file explicitly
|
|
|
|
Another ways to get normal path to %TEMP%
|
|
[Environment]::GetEnvironmentVariable("TEMP", "User")
|
|
(Get-ItemProperty -Path HKCU:\Environment -Name TEMP).TEMP
|
|
[System.IO.Path]::GetTempPath()
|
|
#>
|
|
Get-ChildItem -Path "$DownloadsFolder\VC_redist.x64.exe", "$env:TEMP\dd_vcredist_amd64_*.log" -Force | Remove-Item -Recurse -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message $Localization.NoInternetConnection
|
|
Write-Error -Message $Localization.NoInternetConnection -ErrorAction SilentlyContinue
|
|
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Install the latest .NET Desktop Runtime 6 (x86/x64)
|
|
|
|
.EXAMPLE
|
|
InstallDotNetRuntime6
|
|
|
|
.LINK
|
|
https://docs.microsoft.com/en-us/dotnet/core/install/windows?tabs=net60
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function InstallDotNetRuntime6
|
|
{
|
|
try
|
|
{
|
|
# Check the internet connection
|
|
$Parameters = @{
|
|
Uri = "https://www.google.com"
|
|
Method = "Head"
|
|
DisableKeepAlive = $true
|
|
UseBasicParsing = $true
|
|
}
|
|
if (-not (Invoke-WebRequest @Parameters).StatusDescription)
|
|
{
|
|
return
|
|
}
|
|
|
|
if ([System.Version](Get-AppxPackage -Name Microsoft.DesktopAppInstaller).Version -ge [System.Version]"1.17")
|
|
{
|
|
# .NET Desktop Runtime x86
|
|
winget install --id=Microsoft.DotNet.DesktopRuntime.6 --architecture x86 --exact --accept-source-agreements
|
|
# .NET Desktop Runtime x64
|
|
winget install --id=Microsoft.DotNet.DesktopRuntime.6 --architecture x64 --exact --accept-source-agreements
|
|
}
|
|
else
|
|
{
|
|
# https://github.com/dotnet/core/blob/main/release-notes/releases-index.json
|
|
$Parameters = @{
|
|
Uri = "https://dotnetcli.blob.core.windows.net/dotnet/release-metadata/6.0/releases.json"
|
|
UseBasicParsing = $true
|
|
}
|
|
$LatestRelease = (Invoke-RestMethod @Parameters)."latest-release"
|
|
$DownloadsFolder = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "{374DE290-123F-4565-9164-39C4925E467B}"
|
|
|
|
# .NET Desktop Runtime x86
|
|
$Parameters = @{
|
|
Uri = "https://dotnetcli.azureedge.net/dotnet/Runtime/$LatestRelease/dotnet-runtime-$LatestRelease-win-x64.exe"
|
|
OutFile = "$DownloadsFolder\dotnet-runtime-$LatestRelease-win-x86.exe"
|
|
UseBasicParsing = $true
|
|
Verbose = $true
|
|
}
|
|
Invoke-WebRequest @Parameters
|
|
|
|
Start-Process -FilePath "$DownloadsFolder\dotnet-runtime-$LatestRelease-win-x86.exe" -ArgumentList "/install /passive /norestart" -Wait
|
|
|
|
# .NET Desktop Runtime x64
|
|
$Parameters = @{
|
|
Uri = "https://dotnetcli.azureedge.net/dotnet/Runtime/$LatestRelease/dotnet-runtime-$LatestRelease-win-x86.exe"
|
|
OutFile = "$DownloadsFolder\dotnet-runtime-$LatestRelease-win-x64.exe"
|
|
UseBasicParsing = $true
|
|
Verbose = $true
|
|
}
|
|
Invoke-WebRequest @Parameters
|
|
|
|
Start-Process -FilePath "$DownloadsFolder\dotnet-runtime-$LatestRelease-win-x64.exe" -ArgumentList "/install /passive /norestart" -Wait
|
|
|
|
<#
|
|
PowerShell 5.1 (7.2 too) interprets the 8.3 file name literally, if an environment variable contains a non-latin word,
|
|
so you won't be able to remove "$env:TEMP\Microsoft_Windows_Desktop_Runtime*.log" file explicitly
|
|
|
|
Another ways to get normal path to %TEMP%
|
|
[Environment]::GetEnvironmentVariable("TEMP", "User")
|
|
(Get-ItemProperty -Path HKCU:\Environment -Name TEMP).TEMP
|
|
[System.IO.Path]::GetTempPath()
|
|
#>
|
|
$Paths = @(
|
|
"$DownloadsFolder\dotnet-runtime-$LatestRelease-win-x86.exe",
|
|
"$DownloadsFolder\dotnet-runtime-$LatestRelease-win-x64.exe",
|
|
"$env:TEMP\Microsoft_Windows_Desktop_Runtime*.log"
|
|
)
|
|
Get-ChildItem -Path $Paths -Force -ErrorAction Ignore | Remove-Item -Recurse -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message $Localization.NoInternetConnection
|
|
Write-Error -Message $Localization.NoInternetConnection -ErrorAction SilentlyContinue
|
|
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Bypass RKN restrictins using antizapret.prostovpn.org proxies
|
|
|
|
.PARAMETER Enable
|
|
Enable proxying only blocked sites from the unified registry of Roskomnadzor using antizapret.prostovpn.org servers
|
|
|
|
.PARAMETER Disable
|
|
Disable proxying only blocked sites from the unified registry of Roskomnadzor using antizapret.prostovpn.org servers
|
|
|
|
.EXAMPLE
|
|
RKNBypass -Enable
|
|
|
|
.EXAMPLE
|
|
RKNBypass -Disable
|
|
|
|
.LINK
|
|
https://antizapret.prostovpn.org
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function RKNBypass
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
# If current region is Russia
|
|
if (((Get-WinHomeLocation).GeoId -eq "203"))
|
|
{
|
|
New-ItemProperty -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Internet Settings" -Name AutoConfigURL -PropertyType String -Value "https://antizapret.prostovpn.org/proxy.pac" -Force
|
|
}
|
|
}
|
|
"Disable"
|
|
{
|
|
Remove-ItemProperty -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Internet Settings" -Name AutoConfigURL -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Enable the latest Windows Subsystem for Android™ with Amazon Appstore
|
|
|
|
.EXAMPLE Enable all necessary dependencies (reboot may require) and open Microsoft Store WSA page to install it manually
|
|
WSA -Enable
|
|
|
|
.EXAMPLE Disable all necessary dependencies (reboot may require) and uninstall Windows Subsystem for Android™ with Amazon Appstore
|
|
WSA -Disable
|
|
|
|
.LINK
|
|
https://support.microsoft.com/en-us/windows/install-mobile-apps-and-the-amazon-appstore-f8d0abb5-44ad-47d8-b9fb-ad6b1459ff6c
|
|
|
|
.LINK
|
|
https://docs.microsoft.com/en-us/windows/android/wsa/
|
|
|
|
.LINK
|
|
https://www.microsoft.com/store/productId/9P3395VX91NR
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function WSA
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
# Check if Windows 11 is installed on an SSD
|
|
$DiskNumber = (Get-Disk | Where-Object -FilterScript {$_.Isboot -and $_.IsSystem -and ($_.OperationalStatus -eq "Online")}).Number
|
|
if (Get-PhysicalDisk -DeviceNumber $DiskNumber | Where-Object -FilterScript {$_.MediaType -ne "SSD"})
|
|
{
|
|
Write-Warning -Message $Localization.SSDRequired
|
|
|
|
return
|
|
}
|
|
|
|
# Enable Windows Subsystem for Android (WSA)
|
|
if ((Get-WindowsOptionalFeature -Online -FeatureName Microsoft-Windows-Subsystem-Linux).State -eq "Disabled")
|
|
{
|
|
Enable-WindowsOptionalFeature -Online -FeatureName Microsoft-Windows-Subsystem-Linux -NoRestart
|
|
|
|
Write-Warning -Message $Localization.RestartWarning
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
|
|
return
|
|
}
|
|
|
|
# Enable Virtual Machine Platform
|
|
if ((Get-WindowsOptionalFeature -Online -FeatureName VirtualMachinePlatform).State -eq "Disabled")
|
|
{
|
|
Enable-WindowsOptionalFeature -Online -FeatureName VirtualMachinePlatform -NoRestart
|
|
|
|
Write-Warning -Message $Localization.RestartWarning
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
|
|
return
|
|
}
|
|
|
|
if (Get-AppxPackage -Name MicrosoftCorporationII.WindowsSubsystemForAndroid)
|
|
{
|
|
return
|
|
}
|
|
|
|
try
|
|
{
|
|
# Check the internet connection
|
|
$Parameters = @{
|
|
Uri = "https://www.google.com"
|
|
Method = "Head"
|
|
DisableKeepAlive = $true
|
|
UseBasicParsing = $true
|
|
}
|
|
if (-not (Invoke-WebRequest @Parameters).StatusDescription)
|
|
{
|
|
return
|
|
}
|
|
|
|
if (((Get-WinHomeLocation).GeoId -ne "244"))
|
|
{
|
|
# Set Windows region to USA
|
|
$Script:Region = (Get-WinHomeLocation).GeoId
|
|
Set-WinHomeLocation -GeoId 244
|
|
|
|
$Script:RegionChanged = $true
|
|
}
|
|
|
|
# Open Misrosoft Store WSA page to install it manually
|
|
Start-Process -FilePath ms-windows-store://pdp/?ProductId=9P3395VX91NR
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message $Localization.NoInternetConnection
|
|
Write-Error -Message $Localization.NoInternetConnection -ErrorAction SilentlyContinue
|
|
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
"Disable"
|
|
{
|
|
Disable-WindowsOptionalFeature -Online -FeatureName Microsoft-Windows-Subsystem-Linux -NoRestart
|
|
Get-AppxPackage -Name MicrosoftCorporationII.WindowsSubsystemForAndroid | Remove-AppxPackage
|
|
}
|
|
}
|
|
}
|
|
#endregion System
|
|
|
|
#region WSL
|
|
<#
|
|
.SYNOPSIS
|
|
Windows Subsystem for Linux (WSL)
|
|
|
|
.PARAMETER Install
|
|
Enable Windows Subsystem for Linux (WSL), install the latest WSL Linux kernel version, and a Linux distribution using a pop-up form
|
|
|
|
.EXAMPLE
|
|
WSL
|
|
|
|
.NOTES
|
|
The "Receive updates for other Microsoft products" setting will be enabled automatically to receive kernel updates
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function WSL
|
|
{
|
|
Add-Type -AssemblyName PresentationCore, PresentationFramework
|
|
|
|
#region Variables
|
|
$CommandTag = $null
|
|
#endregion
|
|
|
|
#region Xaml Markup
|
|
[xml]$XAML = '
|
|
<Window
|
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
|
Name="Window"
|
|
Title="WSL"
|
|
MinHeight="400" MinWidth="350"
|
|
SizeToContent="WidthAndHeight" WindowStartupLocation="CenterScreen"
|
|
TextOptions.TextFormattingMode="Display" SnapsToDevicePixels="True"
|
|
FontFamily="Candara" FontSize="16" ShowInTaskbar="True"
|
|
Background="#F1F1F1" Foreground="#262626">
|
|
<Window.Resources>
|
|
<Style TargetType="StackPanel" x:Key="PanelContainerStyle">
|
|
<Setter Property="Grid.Row" Value="0"/>
|
|
<Setter Property="Orientation" Value="Vertical"/>
|
|
</Style>
|
|
<Style TargetType="StackPanel" x:Key="PanelElementStyle">
|
|
<Setter Property="Orientation" Value="Horizontal"/>
|
|
</Style>
|
|
<Style TargetType="RadioButton">
|
|
<Setter Property="VerticalAlignment" Value="Center"/>
|
|
<Setter Property="Margin" Value="10"/>
|
|
</Style>
|
|
<Style TargetType="TextBlock">
|
|
<Setter Property="VerticalAlignment" Value="Center"/>
|
|
<Setter Property="Margin" Value="0, 0, 0, 2"/>
|
|
</Style>
|
|
<Style TargetType="Button">
|
|
<Setter Property="Margin" Value="20"/>
|
|
<Setter Property="Padding" Value="10"/>
|
|
<Setter Property="IsEnabled" Value="False"/>
|
|
</Style>
|
|
</Window.Resources>
|
|
<Grid>
|
|
<Grid.RowDefinitions>
|
|
<RowDefinition Height="*"/>
|
|
<RowDefinition Height="Auto"/>
|
|
</Grid.RowDefinitions>
|
|
<StackPanel Name="PanelContainer" Style="{StaticResource PanelContainerStyle}">
|
|
<StackPanel Style="{StaticResource PanelElementStyle}">
|
|
<RadioButton Name="RadioButtonUbuntu" GroupName="NixNames" Tag="Ubuntu"/>
|
|
<TextBlock Text="Ubuntu"/>
|
|
</StackPanel>
|
|
|
|
<StackPanel Style="{StaticResource PanelElementStyle}">
|
|
<RadioButton Name="RadioButtonDebian" GroupName="NixNames" Tag="Debian"/>
|
|
<TextBlock Text="Debian GNU/Linux"/>
|
|
</StackPanel>
|
|
|
|
<StackPanel Style="{StaticResource PanelElementStyle}">
|
|
<RadioButton Name="RadioButtonKali" GroupName="NixNames" Tag="kali-linux"/>
|
|
<TextBlock Text="Kali Linux Rolling"/>
|
|
</StackPanel>
|
|
|
|
<StackPanel Style="{StaticResource PanelElementStyle}">
|
|
<RadioButton Name="RadioButtonOpenSuse" GroupName="NixNames" Tag="openSUSE-42"/>
|
|
<TextBlock Text="openSUSE Leap 42"/>
|
|
</StackPanel>
|
|
|
|
<StackPanel Style="{StaticResource PanelElementStyle}">
|
|
<RadioButton Name="RadioButtonSuse" GroupName="NixNames" Tag="SLES-12"/>
|
|
<TextBlock Text="SUSE Linux Enterprise Server v12"/>
|
|
</StackPanel>
|
|
|
|
<StackPanel Style="{StaticResource PanelElementStyle}">
|
|
<RadioButton Name="RadioButtonUbuntu16" GroupName="NixNames" Tag="Ubuntu-16.04"/>
|
|
<TextBlock Text="Ubuntu 16.04 LTS"/>
|
|
</StackPanel>
|
|
|
|
<StackPanel Style="{StaticResource PanelElementStyle}">
|
|
<RadioButton Name="RadioButtonUbuntu18" GroupName="NixNames" Tag="Ubuntu-18.04"/>
|
|
<TextBlock Text="Ubuntu 18.04 LTS"/>
|
|
</StackPanel>
|
|
|
|
<StackPanel Style="{StaticResource PanelElementStyle}">
|
|
<RadioButton Name="RadioButtonUbuntu20" GroupName="NixNames" Tag="Ubuntu-20.04"/>
|
|
<TextBlock Text="Ubuntu 20.04 LTS"/>
|
|
</StackPanel>
|
|
</StackPanel>
|
|
<Button Name="ButtonInstall" Content="Install" Grid.Row="2"/>
|
|
</Grid>
|
|
</Window>
|
|
'
|
|
#endregion
|
|
|
|
#region Functions
|
|
function RadioButtonChecked
|
|
{
|
|
$Global:CommandTag = $_.OriginalSource.Tag
|
|
if (-not $ButtonInstall.IsEnabled)
|
|
{
|
|
$ButtonInstall.IsEnabled = $true
|
|
}
|
|
}
|
|
|
|
function ButtonInstallClicked
|
|
{
|
|
Start-Process -FilePath wsl.exe -ArgumentList "--install --distribution $Global:CommandTag" -Wait
|
|
}
|
|
#endregion
|
|
|
|
$Form = [Windows.Markup.XamlReader]::Load((New-Object -TypeName System.Xml.XmlNodeReader -ArgumentList $XAML))
|
|
$XAML.SelectNodes("//*[@*[contains(translate(name(.),'n','N'),'Name')]]") | ForEach-Object -Process {
|
|
$control = $Form.FindName($_.Name)
|
|
Set-Variable -Name ($_.Name) -Value $control
|
|
if ($Control.Template.TargetType.Name -eq "RadioButton")
|
|
{
|
|
$Control.add_Checked({RadioButtonChecked})
|
|
}
|
|
}
|
|
$ButtonInstall.add_Click({ButtonInstallClicked})
|
|
|
|
#region Sendkey function
|
|
# Emulate the Backspace key sending to prevent the console window to freeze
|
|
Start-Sleep -Milliseconds 500
|
|
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
|
|
$SetForegroundWindow = @{
|
|
Namespace = "WinAPI"
|
|
Name = "ForegroundWindow"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
[DllImport("user32.dll")]
|
|
public static extern bool ShowWindowAsync(IntPtr hWnd, int nCmdShow);
|
|
|
|
[DllImport("user32.dll")]
|
|
[return: MarshalAs(UnmanagedType.Bool)]
|
|
public static extern bool SetForegroundWindow(IntPtr hWnd);
|
|
"@
|
|
}
|
|
|
|
if (-not ("WinAPI.ForegroundWindow" -as [type]))
|
|
{
|
|
Add-Type @SetForegroundWindow
|
|
}
|
|
|
|
Get-Process | Where-Object -FilterScript {(($_.ProcessName -eq "powershell") -or ($_.ProcessName -eq "WindowsTerminal")) -and ($_.MainWindowTitle -match "Sophia Script for Windows 11")} | ForEach-Object -Process {
|
|
# Show window, if minimized
|
|
[WinAPI.ForegroundWindow]::ShowWindowAsync($_.MainWindowHandle, 10)
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Force move the console window to the foreground
|
|
[WinAPI.ForegroundWindow]::SetForegroundWindow($_.MainWindowHandle)
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Emulate the Backspace key sending
|
|
[System.Windows.Forms.SendKeys]::SendWait("{BACKSPACE 1}")
|
|
}
|
|
#endregion Sendkey function
|
|
|
|
# Force move the WPF form to the foreground
|
|
$Window.Add_Loaded({$Window.Activate()})
|
|
$Form.ShowDialog() | Out-Null
|
|
|
|
# Receive updates for other Microsoft products when you update Windows
|
|
(New-Object -ComObject Microsoft.Update.ServiceManager).AddService2("7971f918-a847-4430-9279-4a52d1efe18d", 7, "")
|
|
}
|
|
#endregion WSL
|
|
|
|
#region Start menu
|
|
<#
|
|
.SYNOPSIS
|
|
Unpin all Start apps
|
|
|
|
.EXAMPLE
|
|
UnpinAllStartApps
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function UnpinAllStartApps
|
|
{
|
|
# Due to Windows 11 22H2 changes name of start.bin to start2.bin we need to check this first
|
|
if (Test-Path -Path "$env:LOCALAPPDATA\Packages\Microsoft.Windows.StartMenuExperienceHost_cw5n1h2txyewy\LocalState\start.bin")
|
|
{
|
|
$Parameters = @{
|
|
Path = "$PSScriptRoot\..\bin\Start_Layout\start.bin"
|
|
Destination = "$env:LOCALAPPDATA\Packages\Microsoft.Windows.StartMenuExperienceHost_cw5n1h2txyewy\LocalState"
|
|
Force = $true
|
|
}
|
|
Copy-Item @Parameters
|
|
}
|
|
else
|
|
{
|
|
$Parameters = @{
|
|
Path = "$PSScriptRoot\..\bin\Start_Layout\start.bin"
|
|
Destination = "$env:LOCALAPPDATA\Packages\Microsoft.Windows.StartMenuExperienceHost_cw5n1h2txyewy\LocalState"
|
|
Force = $true
|
|
}
|
|
Copy-Item @Parameters
|
|
|
|
Remove-Item -Path "$env:LOCALAPPDATA\Packages\Microsoft.Windows.StartMenuExperienceHost_cw5n1h2txyewy\LocalState\start2.bin" -Force
|
|
$Parameters = @{
|
|
Path = "$env:LOCALAPPDATA\Packages\Microsoft.Windows.StartMenuExperienceHost_cw5n1h2txyewy\LocalState\start.bin"
|
|
NewName = "$env:LOCALAPPDATA\Packages\Microsoft.Windows.StartMenuExperienceHost_cw5n1h2txyewy\LocalState\start2.bin"
|
|
Force = $true
|
|
}
|
|
Rename-Item @Parameters
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
How to run the Windows PowerShell shortcut
|
|
|
|
.PARAMETER Elevated
|
|
Run the Windows PowerShell shortcut from the Start menu as Administrator
|
|
|
|
.PARAMETER NonElevated
|
|
Run the Windows PowerShell shortcut from the Start menu as user
|
|
|
|
.EXAMPLE
|
|
RunPowerShellShortcut -Elevated
|
|
|
|
.EXAMPLE
|
|
RunPowerShellShortcut -NonElevated
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function RunPowerShellShortcut
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Elevated"
|
|
)]
|
|
[switch]
|
|
$Elevated,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "NonElevated"
|
|
)]
|
|
[switch]
|
|
$NonElevated
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Elevated"
|
|
{
|
|
[byte[]]$bytes = Get-Content -Path "$env:APPDATA\Microsoft\Windows\Start Menu\Programs\Windows PowerShell\Windows PowerShell.lnk" -Encoding Byte -Raw
|
|
$bytes[0x15] = $bytes[0x15] -bor 0x20
|
|
Set-Content -Path "$env:APPDATA\Microsoft\Windows\Start Menu\Programs\Windows PowerShell\Windows PowerShell.lnk" -Value $bytes -Encoding Byte -Force
|
|
}
|
|
"NonElevated"
|
|
{
|
|
[byte[]]$bytes = Get-Content -Path "$env:APPDATA\Microsoft\Windows\Start Menu\Programs\Windows PowerShell\Windows PowerShell.lnk" -Encoding Byte -Raw
|
|
$bytes[0x15] = $bytes[0x15] -bxor 0x20
|
|
Set-Content -Path "$env:APPDATA\Microsoft\Windows\Start Menu\Programs\Windows PowerShell\Windows PowerShell.lnk" -Value $bytes -Encoding Byte -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Configure Start layout
|
|
|
|
.PARAMETER Default
|
|
Show default Start layout
|
|
|
|
.PARAMETER ShowMorePins
|
|
Show more pins on Start
|
|
|
|
.PARAMETER ShowMoreRecommendations
|
|
Show more recommendations on Start
|
|
|
|
.EXAMPLE
|
|
StartLayout -Default
|
|
|
|
.EXAMPLE
|
|
StartLayout -ShowMorePins
|
|
|
|
.EXAMPLE
|
|
StartLayout -ShowMoreRecommendations
|
|
|
|
.NOTES
|
|
For Windows 11 Insider Preview 22509+ build only
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function StartLayout
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Default"
|
|
)]
|
|
[switch]
|
|
$Default,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "ShowMorePins"
|
|
)]
|
|
[switch]
|
|
$ShowMorePins,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "ShowMoreRecommendations"
|
|
)]
|
|
[switch]
|
|
$ShowMoreRecommendations
|
|
)
|
|
|
|
if ((Get-CimInstance -ClassName Win32_OperatingSystem).BuildNumber -ge 22509)
|
|
{
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Default"
|
|
{
|
|
# Default
|
|
New-ItemProperty -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\Advanced" -Name Start_Layout -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"ShowMorePins"
|
|
{
|
|
# Show More Pins
|
|
New-ItemProperty -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\Advanced" -Name Start_Layout -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"ShowMoreRecommendations"
|
|
{
|
|
# Show More Recommendations
|
|
New-ItemProperty -Path "HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\Advanced" -Name Start_Layout -PropertyType DWord -Value 2 -Force
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#endregion Start menu
|
|
|
|
#region UWP apps
|
|
<#
|
|
.SYNOPSIS
|
|
Uninstall UWP apps
|
|
|
|
.PARAMETER ForAllUsers
|
|
The "ForAllUsers" argument sets a checkbox to unistall packages for all users
|
|
|
|
.EXAMPLE
|
|
UninstallUWPApps
|
|
|
|
.EXAMPLE
|
|
UninstallUWPApps -ForAllUsers
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function UninstallUWPApps
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(Mandatory = $false)]
|
|
[switch]
|
|
$ForAllUsers
|
|
)
|
|
|
|
Add-Type -AssemblyName PresentationCore, PresentationFramework
|
|
|
|
#region Variables
|
|
# The following UWP apps will have their checkboxes unchecked
|
|
$UncheckedAppxPackages = @(
|
|
# AMD Radeon Software
|
|
"AdvancedMicroDevicesInc-2.AMDRadeonSoftware",
|
|
|
|
# Intel Graphics Control Center
|
|
"AppUp.IntelGraphicsControlPanel",
|
|
"AppUp.IntelGraphicsExperience",
|
|
|
|
# Sticky Notes
|
|
"Microsoft.MicrosoftStickyNotes",
|
|
|
|
# Screen Sketch
|
|
"Microsoft.ScreenSketch",
|
|
|
|
# Photos (and Video Editor)
|
|
"Microsoft.Windows.Photos",
|
|
"Microsoft.Photos.MediaEngineDLC",
|
|
"Microsoft.RawImageExtension"
|
|
|
|
# HEVC Video Extensions from Device Manufacturer
|
|
"Microsoft.HEVCVideoExtension",
|
|
|
|
# Calculator
|
|
"Microsoft.WindowsCalculator",
|
|
|
|
# Windows Camera
|
|
"Microsoft.WindowsCamera",
|
|
|
|
# Xbox Identity Provider
|
|
"Microsoft.XboxIdentityProvider",
|
|
|
|
# Xbox Console Companion
|
|
"Microsoft.XboxApp",
|
|
|
|
# Xbox
|
|
"Microsoft.GamingApp",
|
|
"Microsoft.GamingServices",
|
|
|
|
# Paint
|
|
"Microsoft.Paint",
|
|
|
|
# Xbox TCUI
|
|
"Microsoft.Xbox.TCUI",
|
|
|
|
# Xbox Speech To Text Overlay
|
|
"Microsoft.XboxSpeechToTextOverlay",
|
|
|
|
# Xbox Game Bar
|
|
"Microsoft.XboxGamingOverlay",
|
|
|
|
# Xbox Game Bar Plugin
|
|
"Microsoft.XboxGameOverlay",
|
|
|
|
# NVIDIA Control Panel
|
|
"NVIDIACorp.NVIDIAControlPanel",
|
|
|
|
# Realtek Audio Console
|
|
"RealtekSemiconductorCorp.RealtekAudioControl"
|
|
)
|
|
|
|
# The following UWP apps will be excluded from the display
|
|
$ExcludedAppxPackages = @(
|
|
# Microsoft Desktop App Installer
|
|
"Microsoft.DesktopAppInstaller",
|
|
|
|
# Store Experience Host
|
|
"Microsoft.StorePurchaseApp",
|
|
|
|
# Notepad
|
|
"Microsoft.WindowsNotepad",
|
|
|
|
# Microsoft Store
|
|
"Microsoft.WindowsStore",
|
|
|
|
# Windows Terminal
|
|
"Microsoft.WindowsTerminal",
|
|
"Microsoft.WindowsTerminalPreview",
|
|
|
|
# Web Media Extensions
|
|
"Microsoft.WebMediaExtensions",
|
|
|
|
# AV1 Video Extension
|
|
"Microsoft.AV1VideoExtension",
|
|
|
|
# HEVC Video Extensions from Device Manufacturer
|
|
"Microsoft.HEVCVideoExtension"
|
|
)
|
|
|
|
#region Variables
|
|
#region XAML Markup
|
|
# The section defines the design of the upcoming dialog box
|
|
[xml]$XAML = '
|
|
<Window
|
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
|
Name="Window"
|
|
MinHeight="400" MinWidth="415"
|
|
SizeToContent="Width" WindowStartupLocation="CenterScreen"
|
|
TextOptions.TextFormattingMode="Display" SnapsToDevicePixels="True"
|
|
FontFamily="Candara" FontSize="16" ShowInTaskbar="True"
|
|
Background="#F1F1F1" Foreground="#262626">
|
|
<Window.Resources>
|
|
<Style TargetType="StackPanel">
|
|
<Setter Property="Orientation" Value="Horizontal"/>
|
|
<Setter Property="VerticalAlignment" Value="Top"/>
|
|
</Style>
|
|
<Style TargetType="CheckBox">
|
|
<Setter Property="Margin" Value="10, 13, 10, 10"/>
|
|
<Setter Property="IsChecked" Value="True"/>
|
|
</Style>
|
|
<Style TargetType="TextBlock">
|
|
<Setter Property="Margin" Value="0, 10, 10, 10"/>
|
|
</Style>
|
|
<Style TargetType="Button">
|
|
<Setter Property="Margin" Value="20"/>
|
|
<Setter Property="Padding" Value="10"/>
|
|
<Setter Property="IsEnabled" Value="False"/>
|
|
</Style>
|
|
<Style TargetType="Border">
|
|
<Setter Property="Grid.Row" Value="1"/>
|
|
<Setter Property="CornerRadius" Value="0"/>
|
|
<Setter Property="BorderThickness" Value="0, 1, 0, 1"/>
|
|
<Setter Property="BorderBrush" Value="#000000"/>
|
|
</Style>
|
|
<Style TargetType="ScrollViewer">
|
|
<Setter Property="HorizontalScrollBarVisibility" Value="Disabled"/>
|
|
<Setter Property="BorderBrush" Value="#000000"/>
|
|
<Setter Property="BorderThickness" Value="0, 1, 0, 1"/>
|
|
</Style>
|
|
</Window.Resources>
|
|
<Grid>
|
|
<Grid.RowDefinitions>
|
|
<RowDefinition Height="Auto"/>
|
|
<RowDefinition Height="*"/>
|
|
<RowDefinition Height="Auto"/>
|
|
</Grid.RowDefinitions>
|
|
<Grid Grid.Row="0">
|
|
<Grid.ColumnDefinitions>
|
|
<ColumnDefinition Width="*"/>
|
|
<ColumnDefinition Width="*"/>
|
|
</Grid.ColumnDefinitions>
|
|
<StackPanel Name="PanelSelectAll" Grid.Column="0" HorizontalAlignment="Left">
|
|
<CheckBox Name="CheckBoxSelectAll" IsChecked="False"/>
|
|
<TextBlock Name="TextBlockSelectAll" Margin="10,10, 0, 10"/>
|
|
</StackPanel>
|
|
<StackPanel Name="PanelRemoveForAll" Grid.Column="1" HorizontalAlignment="Right">
|
|
<TextBlock Name="TextBlockRemoveForAll" Margin="10,10, 0, 10"/>
|
|
<CheckBox Name="CheckBoxForAllUsers" IsChecked="False"/>
|
|
</StackPanel>
|
|
</Grid>
|
|
<Border>
|
|
<ScrollViewer>
|
|
<StackPanel Name="PanelContainer" Orientation="Vertical"/>
|
|
</ScrollViewer>
|
|
</Border>
|
|
<Button Name="ButtonUninstall" Grid.Row="2"/>
|
|
</Grid>
|
|
</Window>
|
|
'
|
|
#endregion XAML Markup
|
|
|
|
$Reader = (New-Object -TypeName System.Xml.XmlNodeReader -ArgumentList $XAML)
|
|
$Form = [Windows.Markup.XamlReader]::Load($Reader)
|
|
$XAML.SelectNodes("//*[@*[contains(translate(name(.),'n','N'),'Name')]]") | ForEach-Object -Process {
|
|
Set-Variable -Name ($_.Name) -Value $Form.FindName($_.Name)
|
|
}
|
|
|
|
$Window.Title = $Localization.UWPAppsTitle
|
|
$ButtonUninstall.Content = $Localization.Uninstall
|
|
$TextBlockRemoveForAll.Text = $Localization.UninstallUWPForAll
|
|
$TextBlockSelectAll.Text = $Localization.SelectAll
|
|
|
|
$ButtonUninstall.Add_Click({ButtonUninstallClick})
|
|
$CheckBoxForAllUsers.Add_Click({CheckBoxForAllUsersClick})
|
|
$CheckBoxSelectAll.Add_Click({CheckBoxSelectAllClick})
|
|
#endregion Variables
|
|
|
|
#region Functions
|
|
function Get-AppxBundle
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[string[]]
|
|
$Exclude,
|
|
|
|
[switch]
|
|
$AllUsers
|
|
)
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
$AppxPackages = @(Get-AppxPackage -PackageTypeFilter Bundle -AllUsers:$AllUsers | Where-Object -FilterScript {$_.Name -notin $ExcludedAppxPackages})
|
|
|
|
# The Bundle packages contains no Microsoft Teams
|
|
if (Get-AppxPackage -Name MicrosoftTeams -AllUsers:$AllUsers)
|
|
{
|
|
# Temporarily hack: due to the fact that there are actually two Microsoft Teams packages, we need to choose the first one to display
|
|
$AppxPackages += Get-AppxPackage -Name MicrosoftTeams -AllUsers:$AllUsers | Select-Object -Index 0
|
|
}
|
|
|
|
# The Bundle packages contains no Spotify
|
|
if (Get-AppxPackage -Name SpotifyAB.SpotifyMusic -AllUsers:$AllUsers)
|
|
{
|
|
# Temporarily hack: due to the fact that there are actually two Microsoft Teams packages, we need to choose the first one to display
|
|
$AppxPackages += Get-AppxPackage -Name SpotifyAB.SpotifyMusic -AllUsers:$AllUsers | Select-Object -Index 0
|
|
}
|
|
|
|
$PackagesIds = [Windows.Management.Deployment.PackageManager, Windows.Web, ContentType = WindowsRuntime]::new().FindPackages() | Select-Object -Property DisplayName -ExpandProperty Id | Select-Object -Property Name, DisplayName
|
|
|
|
foreach ($AppxPackage in $AppxPackages)
|
|
{
|
|
$PackageId = $PackagesIds | Where-Object -FilterScript {$_.Name -eq $AppxPackage.Name}
|
|
|
|
if (-not $PackageId)
|
|
{
|
|
continue
|
|
}
|
|
|
|
[PSCustomObject]@{
|
|
Name = $AppxPackage.Name
|
|
PackageFullName = $AppxPackage.PackageFullName
|
|
DisplayName = $PackageId.DisplayName
|
|
}
|
|
}
|
|
}
|
|
|
|
function Add-Control
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ValueFromPipeline = $true
|
|
)]
|
|
[ValidateNotNull()]
|
|
[PSCustomObject[]]
|
|
$Packages
|
|
)
|
|
|
|
process
|
|
{
|
|
foreach ($Package in $Packages)
|
|
{
|
|
$CheckBox = New-Object -TypeName System.Windows.Controls.CheckBox
|
|
$CheckBox.Tag = $Package.PackageFullName
|
|
|
|
$TextBlock = New-Object -TypeName System.Windows.Controls.TextBlock
|
|
|
|
if ($Package.DisplayName)
|
|
{
|
|
$TextBlock.Text = $Package.DisplayName
|
|
}
|
|
else
|
|
{
|
|
$TextBlock.Text = $Package.Name
|
|
}
|
|
|
|
$StackPanel = New-Object -TypeName System.Windows.Controls.StackPanel
|
|
$StackPanel.Children.Add($CheckBox) | Out-Null
|
|
$StackPanel.Children.Add($TextBlock) | Out-Null
|
|
|
|
$PanelContainer.Children.Add($StackPanel) | Out-Null
|
|
|
|
if ($UncheckedAppxPackages.Contains($Package.Name))
|
|
{
|
|
$CheckBox.IsChecked = $false
|
|
}
|
|
else
|
|
{
|
|
$CheckBox.IsChecked = $true
|
|
$PackagesToRemove.Add($Package.PackageFullName)
|
|
}
|
|
|
|
$CheckBox.Add_Click({CheckBoxClick})
|
|
}
|
|
}
|
|
}
|
|
|
|
function CheckBoxForAllUsersClick
|
|
{
|
|
$PanelContainer.Children.RemoveRange(0, $PanelContainer.Children.Count)
|
|
$PackagesToRemove.Clear()
|
|
$AppXPackages = Get-AppxBundle -Exclude $ExcludedAppxPackages -AllUsers:$CheckBoxForAllUsers.IsChecked
|
|
$AppXPackages | Add-Control
|
|
|
|
ButtonUninstallSetIsEnabled
|
|
}
|
|
|
|
function ButtonUninstallClick
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
$Window.Close() | Out-Null
|
|
|
|
# If Xbox Game Bar is selected to uninstall stop its' processes
|
|
if ($PackagesToRemove -match "Microsoft.XboxGamingOverlay")
|
|
{
|
|
Get-Process -Name GameBar, GameBarFTServer -ErrorAction Ignore | Stop-Process -Force
|
|
}
|
|
|
|
$PackagesToRemove | Remove-AppxPackage -AllUsers:$CheckBoxForAllUsers.IsChecked -Verbose
|
|
}
|
|
|
|
function CheckBoxClick
|
|
{
|
|
$CheckBox = $_.Source
|
|
|
|
if ($CheckBox.IsChecked)
|
|
{
|
|
$PackagesToRemove.Add($CheckBox.Tag) | Out-Null
|
|
}
|
|
else
|
|
{
|
|
$PackagesToRemove.Remove($CheckBox.Tag)
|
|
}
|
|
|
|
ButtonUninstallSetIsEnabled
|
|
}
|
|
|
|
function CheckBoxSelectAllClick
|
|
{
|
|
$CheckBox = $_.Source
|
|
|
|
if ($CheckBox.IsChecked)
|
|
{
|
|
$PackagesToRemove.Clear()
|
|
|
|
foreach ($Item in $PanelContainer.Children.Children)
|
|
{
|
|
if ($Item -is [System.Windows.Controls.CheckBox])
|
|
{
|
|
$Item.IsChecked = $true
|
|
$PackagesToRemove.Add($Item.Tag)
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
$PackagesToRemove.Clear()
|
|
|
|
foreach ($Item in $PanelContainer.Children.Children)
|
|
{
|
|
if ($Item -is [System.Windows.Controls.CheckBox])
|
|
{
|
|
$Item.IsChecked = $false
|
|
}
|
|
}
|
|
}
|
|
|
|
ButtonUninstallSetIsEnabled
|
|
}
|
|
|
|
function ButtonUninstallSetIsEnabled
|
|
{
|
|
if ($PackagesToRemove.Count -gt 0)
|
|
{
|
|
$ButtonUninstall.IsEnabled = $true
|
|
}
|
|
else
|
|
{
|
|
$ButtonUninstall.IsEnabled = $false
|
|
}
|
|
}
|
|
#endregion Functions
|
|
|
|
# Check "For all users" checkbox to uninstall packages from all accounts
|
|
if ($ForAllUsers)
|
|
{
|
|
$CheckBoxForAllUsers.IsChecked = $true
|
|
}
|
|
|
|
$PackagesToRemove = [Collections.Generic.List[string]]::new()
|
|
$AppXPackages = Get-AppxBundle -Exclude $ExcludedAppxPackages -AllUsers:$ForAllUsers
|
|
$AppXPackages | Add-Control
|
|
|
|
if ($AppxPackages.Count -eq 0)
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.NoData -Verbose
|
|
}
|
|
else
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.DialogBoxOpening -Verbose
|
|
|
|
#region Sendkey function
|
|
# Emulate the Backspace key sending to prevent the console window to freeze
|
|
Start-Sleep -Milliseconds 500
|
|
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
|
|
$SetForegroundWindow = @{
|
|
Namespace = "WinAPI"
|
|
Name = "ForegroundWindow"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
[DllImport("user32.dll")]
|
|
public static extern bool ShowWindowAsync(IntPtr hWnd, int nCmdShow);
|
|
|
|
[DllImport("user32.dll")]
|
|
[return: MarshalAs(UnmanagedType.Bool)]
|
|
public static extern bool SetForegroundWindow(IntPtr hWnd);
|
|
"@
|
|
}
|
|
|
|
if (-not ("WinAPI.ForegroundWindow" -as [type]))
|
|
{
|
|
Add-Type @SetForegroundWindow
|
|
}
|
|
|
|
Get-Process | Where-Object -FilterScript {(($_.ProcessName -eq "powershell") -or ($_.ProcessName -eq "WindowsTerminal")) -and ($_.MainWindowTitle -match "Sophia Script for Windows 11")} | ForEach-Object -Process {
|
|
# Show window, if minimized
|
|
[WinAPI.ForegroundWindow]::ShowWindowAsync($_.MainWindowHandle, 10)
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Force move the console window to the foreground
|
|
[WinAPI.ForegroundWindow]::SetForegroundWindow($_.MainWindowHandle)
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Emulate the Backspace key sending to prevent the console window to freeze
|
|
[System.Windows.Forms.SendKeys]::SendWait("{BACKSPACE 1}")
|
|
}
|
|
#endregion Sendkey function
|
|
|
|
if ($PackagesToRemove.Count -gt 0)
|
|
{
|
|
$ButtonUninstall.IsEnabled = $true
|
|
}
|
|
|
|
# Force move the WPF form to the foreground
|
|
$Window.Add_Loaded({$Window.Activate()})
|
|
$Form.ShowDialog() | Out-Null
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Restore the default UWP apps
|
|
|
|
.EXAMPLE
|
|
RestoreUWPAppsUWPApps
|
|
|
|
.NOTES
|
|
UWP apps can be restored only if they were uninstalled for the current user
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function RestoreUWPApps
|
|
{
|
|
Add-Type -AssemblyName PresentationCore, PresentationFramework
|
|
|
|
#region Variables
|
|
#region XAML Markup
|
|
# The section defines the design of the upcoming dialog box
|
|
[xml]$XAML = '
|
|
<Window
|
|
xmlns="http://schemas.microsoft.com/winfx/2006/xaml/presentation"
|
|
xmlns:x="http://schemas.microsoft.com/winfx/2006/xaml"
|
|
Name="Window"
|
|
MinHeight="400" MinWidth="410"
|
|
SizeToContent="Width" WindowStartupLocation="CenterScreen"
|
|
TextOptions.TextFormattingMode="Display" SnapsToDevicePixels="True"
|
|
FontFamily="Candara" FontSize="16" ShowInTaskbar="True"
|
|
Background="#F1F1F1" Foreground="#262626">
|
|
<Window.Resources>
|
|
<Style TargetType="StackPanel">
|
|
<Setter Property="Orientation" Value="Horizontal"/>
|
|
<Setter Property="VerticalAlignment" Value="Top"/>
|
|
</Style>
|
|
<Style TargetType="CheckBox">
|
|
<Setter Property="Margin" Value="10, 13, 10, 10"/>
|
|
<Setter Property="IsChecked" Value="True"/>
|
|
</Style>
|
|
<Style TargetType="TextBlock">
|
|
<Setter Property="Margin" Value="0, 10, 10, 10"/>
|
|
</Style>
|
|
<Style TargetType="Button">
|
|
<Setter Property="Margin" Value="20"/>
|
|
<Setter Property="Padding" Value="10"/>
|
|
<Setter Property="IsEnabled" Value="False"/>
|
|
</Style>
|
|
<Style TargetType="Border">
|
|
<Setter Property="Grid.Row" Value="1"/>
|
|
<Setter Property="CornerRadius" Value="0"/>
|
|
<Setter Property="BorderThickness" Value="0, 1, 0, 1"/>
|
|
<Setter Property="BorderBrush" Value="#000000"/>
|
|
</Style>
|
|
<Style TargetType="ScrollViewer">
|
|
<Setter Property="HorizontalScrollBarVisibility" Value="Disabled"/>
|
|
<Setter Property="BorderBrush" Value="#000000"/>
|
|
<Setter Property="BorderThickness" Value="0, 1, 0, 1"/>
|
|
</Style>
|
|
</Window.Resources>
|
|
<Grid>
|
|
<Grid.RowDefinitions>
|
|
<RowDefinition Height="Auto"/>
|
|
<RowDefinition Height="*"/>
|
|
<RowDefinition Height="Auto"/>
|
|
</Grid.RowDefinitions>
|
|
<Grid Grid.Row="0">
|
|
<Grid.ColumnDefinitions>
|
|
<ColumnDefinition Width="*"/>
|
|
<ColumnDefinition Width="*"/>
|
|
</Grid.ColumnDefinitions>
|
|
<StackPanel Name="PanelSelectAll" Grid.Column="0" HorizontalAlignment="Left">
|
|
<CheckBox Name="CheckBoxSelectAll" IsChecked="False"/>
|
|
<TextBlock Name="TextBlockSelectAll" Margin="10,10, 0, 10"/>
|
|
</StackPanel>
|
|
</Grid>
|
|
<Border>
|
|
<ScrollViewer>
|
|
<StackPanel Name="PanelContainer" Orientation="Vertical"/>
|
|
</ScrollViewer>
|
|
</Border>
|
|
<Button Name="ButtonRestore" Grid.Row="2"/>
|
|
</Grid>
|
|
</Window>
|
|
'
|
|
#endregion XAML Markup
|
|
|
|
$Reader = (New-Object -TypeName System.Xml.XmlNodeReader -ArgumentList $XAML)
|
|
$Form = [Windows.Markup.XamlReader]::Load($Reader)
|
|
$XAML.SelectNodes("//*[@*[contains(translate(name(.),'n','N'),'Name')]]") | ForEach-Object -Process {
|
|
Set-Variable -Name ($_.Name) -Value $Form.FindName($_.Name)
|
|
}
|
|
|
|
$Window.Title = $Localization.UWPAppsTitle
|
|
$ButtonRestore.Content = $Localization.Restore
|
|
$TextBlockSelectAll.Text = $Localization.SelectAll
|
|
|
|
$ButtonRestore.Add_Click({ButtonRestoreClick})
|
|
$CheckBoxSelectAll.Add_Click({CheckBoxSelectAllClick})
|
|
#endregion Variables
|
|
|
|
#region Functions
|
|
function Get-AppxManifest
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
# You cannot retrieve packages using -PackageTypeFilter Bundle, otherwise you won't get the InstallLocation attribute. It can be retrieved only by comparing with $Bundles
|
|
$Bundles = (Get-AppXPackage -PackageTypeFilter Bundle -AllUsers).Name
|
|
$AppxPackages = @(Get-AppxPackage -AllUsers | Where-Object -FilterScript {$_.PackageUserInformation -match "Staged"} | Where-Object -FilterScript {$_.Name -in $Bundles})
|
|
|
|
# The Bundle packages contains no Microsoft Teams
|
|
if (Get-AppxPackage -Name MicrosoftTeams -AllUsers)
|
|
{
|
|
# Temporarily hack: due to the fact that there are actually two Microsoft Teams packages, we need to choose the first one to display
|
|
$AppxPackages += Get-AppxPackage -Name MicrosoftTeams -AllUsers | Where-Object -FilterScript {$_.PackageUserInformation -match "Staged"} | Select-Object -Index 0
|
|
}
|
|
|
|
# The Bundle packages contains no Microsoft Teams
|
|
if (Get-AppxPackage -Name SpotifyAB.SpotifyMusic -AllUsers)
|
|
{
|
|
# Temporarily hack: due to the fact that there are actually two Spotify packages, we need to choose the first one to display
|
|
$AppxPackages += Get-AppxPackage -Name SpotifyAB.SpotifyMusic -AllUsers | Where-Object -FilterScript {$_.PackageUserInformation -match "Staged"} | Select-Object -Index 0
|
|
}
|
|
|
|
$PackagesIds = [Windows.Management.Deployment.PackageManager, Windows.Web, ContentType = WindowsRuntime]::new().FindPackages() | Select-Object -Property DisplayName -ExpandProperty Id | Select-Object -Property Name, DisplayName
|
|
|
|
foreach ($AppxPackage in $AppxPackages)
|
|
{
|
|
$PackageId = $PackagesIds | Where-Object -FilterScript {$_.Name -eq $AppxPackage.Name}
|
|
|
|
if (-not $PackageId)
|
|
{
|
|
continue
|
|
}
|
|
|
|
[PSCustomObject]@{
|
|
Name = $AppxPackage.Name
|
|
PackageFullName = $AppxPackage.PackageFullName
|
|
DisplayName = $PackageId.DisplayName
|
|
AppxManifest = "$($AppxPackage.InstallLocation)\AppxManifest.xml"
|
|
}
|
|
}
|
|
}
|
|
|
|
function Add-Control
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ValueFromPipeline = $true
|
|
)]
|
|
[ValidateNotNull()]
|
|
[PSCustomObject[]]
|
|
$Packages
|
|
)
|
|
|
|
process
|
|
{
|
|
foreach ($Package in $Packages)
|
|
{
|
|
$CheckBox = New-Object -TypeName System.Windows.Controls.CheckBox
|
|
$CheckBox.Tag = $Package.AppxManifest
|
|
|
|
$TextBlock = New-Object -TypeName System.Windows.Controls.TextBlock
|
|
|
|
if ($Package.DisplayName)
|
|
{
|
|
$TextBlock.Text = $Package.DisplayName
|
|
}
|
|
else
|
|
{
|
|
$TextBlock.Text = $Package.Name
|
|
}
|
|
|
|
$StackPanel = New-Object -TypeName System.Windows.Controls.StackPanel
|
|
$StackPanel.Children.Add($CheckBox) | Out-Null
|
|
$StackPanel.Children.Add($TextBlock) | Out-Null
|
|
|
|
$PanelContainer.Children.Add($StackPanel) | Out-Null
|
|
|
|
$CheckBox.IsChecked = $true
|
|
$PackagesToRestore.Add($Package.AppxManifest)
|
|
|
|
$CheckBox.Add_Click({CheckBoxClick})
|
|
}
|
|
}
|
|
}
|
|
|
|
function ButtonRestoreClick
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
$Window.Close() | Out-Null
|
|
|
|
$Parameters = @{
|
|
Register = $true
|
|
ForceApplicationShutdown = $true
|
|
ForceUpdateFromAnyVersion = $true
|
|
DisableDevelopmentMode = $true
|
|
Verbose = $true
|
|
}
|
|
$PackagesToRestore | Add-AppxPackage @Parameters
|
|
}
|
|
|
|
function CheckBoxClick
|
|
{
|
|
$CheckBox = $_.Source
|
|
|
|
if ($CheckBox.IsChecked)
|
|
{
|
|
$PackagesToRestore.Add($CheckBox.Tag) | Out-Null
|
|
}
|
|
else
|
|
{
|
|
$PackagesToRestore.Remove($CheckBox.Tag)
|
|
}
|
|
|
|
ButtonRestoreSetIsEnabled
|
|
}
|
|
|
|
function CheckBoxSelectAllClick
|
|
{
|
|
$CheckBox = $_.Source
|
|
|
|
if ($CheckBox.IsChecked)
|
|
{
|
|
$PackagesToRestore.Clear()
|
|
|
|
foreach ($Item in $PanelContainer.Children.Children)
|
|
{
|
|
if ($Item -is [System.Windows.Controls.CheckBox])
|
|
{
|
|
$Item.IsChecked = $true
|
|
$PackagesToRestore.Add($Item.Tag)
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
$PackagesToRestore.Clear()
|
|
|
|
foreach ($Item in $PanelContainer.Children.Children)
|
|
{
|
|
if ($Item -is [System.Windows.Controls.CheckBox])
|
|
{
|
|
$Item.IsChecked = $false
|
|
}
|
|
}
|
|
}
|
|
|
|
ButtonRestoreSetIsEnabled
|
|
}
|
|
|
|
function ButtonRestoreSetIsEnabled
|
|
{
|
|
if ($PackagesToRestore.Count -gt 0)
|
|
{
|
|
$ButtonRestore.IsEnabled = $true
|
|
}
|
|
else
|
|
{
|
|
$ButtonRestore.IsEnabled = $false
|
|
}
|
|
}
|
|
#endregion Functions
|
|
|
|
$PackagesToRestore = [Collections.Generic.List[string]]::new()
|
|
$AppXPackages = Get-AppxManifest
|
|
$AppXPackages | Add-Control
|
|
|
|
if ($AppxPackages.Count -eq 0)
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.NoData -Verbose
|
|
}
|
|
else
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.DialogBoxOpening -Verbose
|
|
|
|
#region Sendkey function
|
|
# Emulate the Backspace key sending to prevent the console window to freeze
|
|
Start-Sleep -Milliseconds 500
|
|
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
|
|
$SetForegroundWindow = @{
|
|
Namespace = "WinAPI"
|
|
Name = "ForegroundWindow"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
[DllImport("user32.dll")]
|
|
public static extern bool ShowWindowAsync(IntPtr hWnd, int nCmdShow);
|
|
|
|
[DllImport("user32.dll")]
|
|
[return: MarshalAs(UnmanagedType.Bool)]
|
|
public static extern bool SetForegroundWindow(IntPtr hWnd);
|
|
"@
|
|
}
|
|
|
|
if (-not ("WinAPI.ForegroundWindow" -as [type]))
|
|
{
|
|
Add-Type @SetForegroundWindow
|
|
}
|
|
|
|
Get-Process | Where-Object -FilterScript {(($_.ProcessName -eq "powershell") -or ($_.ProcessName -eq "WindowsTerminal")) -and ($_.MainWindowTitle -match "Sophia Script for Windows 11")} | ForEach-Object -Process {
|
|
# Show window, if minimized
|
|
[WinAPI.ForegroundWindow]::ShowWindowAsync($_.MainWindowHandle, 10)
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Force move the console window to the foreground
|
|
[WinAPI.ForegroundWindow]::SetForegroundWindow($_.MainWindowHandle)
|
|
|
|
Start-Sleep -Seconds 1
|
|
|
|
# Emulate the Backspace key sending to prevent the console window to freeze
|
|
[System.Windows.Forms.SendKeys]::SendWait("{BACKSPACE 1}")
|
|
}
|
|
#endregion Sendkey function
|
|
|
|
if ($PackagesToRestore.Count -gt 0)
|
|
{
|
|
$ButtonRestore.IsEnabled = $true
|
|
}
|
|
|
|
# Force move the WPF form to the foreground
|
|
$Window.Add_Loaded({$Window.Activate()})
|
|
$Form.ShowDialog() | Out-Null
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Install "HEVC Video Extensions from Device Manufacturer" to be able to open .heic and .heif formats
|
|
|
|
.PARAMETER Install
|
|
Download and install the "HEVC Video Extensions from Device Manufacturer" extension
|
|
|
|
.PARAMETER Manually
|
|
Open Microsoft Store "HEVC Video Extensions from Device Manufacturer" page to install the extension manually
|
|
|
|
.EXAMPLE
|
|
HEIF -Install
|
|
|
|
.EXAMPLE
|
|
HEIF -Manually
|
|
|
|
.LINK
|
|
https://www.microsoft.com/store/productId/9n4wgh0z6vhq
|
|
|
|
.NOTES
|
|
The extension can be installed without Microsoft account
|
|
|
|
.NOTES
|
|
HEVC Video Extension is already installed in Windows 11 22H2 by default
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function HEIF
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Install"
|
|
)]
|
|
[switch]
|
|
$Install,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Manually"
|
|
)]
|
|
[switch]
|
|
$Manually
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Install"
|
|
{
|
|
# Check whether the extension is already installed
|
|
if ((Get-AppxPackage -Name Microsoft.HEVCVideoExtension) -and (Get-AppxPackage -Name Microsoft.Windows.Photos))
|
|
{
|
|
return
|
|
}
|
|
|
|
try
|
|
{
|
|
# Check the internet connection
|
|
$Parameters = @{
|
|
Uri = "https://www.google.com"
|
|
Method = "Head"
|
|
DisableKeepAlive = $true
|
|
UseBasicParsing = $true
|
|
}
|
|
if (-not (Invoke-WebRequest @Parameters).StatusDescription)
|
|
{
|
|
return
|
|
}
|
|
|
|
try
|
|
{
|
|
# Check whether https://github.com is alive
|
|
$Parameters = @{
|
|
Uri = "https://github.com"
|
|
Method = "Head"
|
|
DisableKeepAlive = $true
|
|
UseBasicParsing = $true
|
|
}
|
|
if (-not (Invoke-WebRequest @Parameters).StatusDescription)
|
|
{
|
|
return
|
|
}
|
|
|
|
# https://github.com/Sophia-Community/SophiApp/tree/master/AppX
|
|
$DownloadsFolder = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "{374DE290-123F-4565-9164-39C4925E467B}"
|
|
$Parameters = @{
|
|
Uri = "https://github.com/Sophia-Community/SophiApp/raw/master/AppX/Microsoft.HEVCVideoExtension_2.0.51121.0_x64__8wekyb3d8bbwe.Appx"
|
|
OutFile = "$DownloadsFolder\Microsoft.HEVCVideoExtension_2.0.51121.0_x64__8wekyb3d8bbwe.Appx"
|
|
UseBasicParsing = $true
|
|
Verbose = $true
|
|
}
|
|
Invoke-WebRequest @Parameters
|
|
|
|
# Installing "HEVC Video Extensions from Device Manufacturer"
|
|
Add-AppxPackage -Path "$DownloadsFolder\Microsoft.HEVCVideoExtension_2.0.51121.0_x64__8wekyb3d8bbwe.Appx" -Verbose
|
|
|
|
Remove-Item -Path "$DownloadsFolder\Microsoft.HEVCVideoExtension_2.0.51121.0_x64__8wekyb3d8bbwe.Appx" -Force
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message ($Localization.NoResponse -f "https://github.com")
|
|
Write-Error -Message ($Localization.NoResponse -f "https://github.com") -ErrorAction SilentlyContinue
|
|
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message $Localization.NoInternetConnection
|
|
Write-Error -Message $Localization.NoInternetConnection -ErrorAction SilentlyContinue
|
|
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
"Manually"
|
|
{
|
|
if ((-not (Get-AppxPackage -Name Microsoft.HEVCVideoExtension)) -and (Get-AppxPackage -Name Microsoft.Windows.Photos))
|
|
{
|
|
try
|
|
{
|
|
# Check the internet connection
|
|
$Parameters = @{
|
|
Uri = "https://www.google.com"
|
|
Method = "Head"
|
|
DisableKeepAlive = $true
|
|
UseBasicParsing = $true
|
|
}
|
|
if (-not (Invoke-WebRequest @Parameters).StatusDescription)
|
|
{
|
|
return
|
|
}
|
|
|
|
Start-Process -FilePath ms-windows-store://pdp/?ProductId=9n4wgh0z6vhq
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message $Localization.NoInternetConnection
|
|
Write-Error -Message $Localization.NoInternetConnection -ErrorAction SilentlyContinue
|
|
|
|
Write-Error -Message ($Localization.RestartFunction -f $MyInvocation.Line) -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Cortana autostarting
|
|
|
|
.PARAMETER Disable
|
|
Disable Cortana autostarting
|
|
|
|
.PARAMETER Enable
|
|
Enable Cortana autostarting
|
|
|
|
.EXAMPLE
|
|
CortanaAutostart -Disable
|
|
|
|
.EXAMPLE
|
|
CortanaAutostart -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function CortanaAutostart
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.549981C3F5F10)
|
|
{
|
|
if (-not (Test-Path -Path "Registry::HKEY_CLASSES_ROOT\Local Settings\Software\Microsoft\Windows\CurrentVersion\AppModel\SystemAppData\Microsoft.549981C3F5F10_8wekyb3d8bbwe\CortanaStartupId"))
|
|
{
|
|
New-Item -Path "Registry::HKEY_CLASSES_ROOT\Local Settings\Software\Microsoft\Windows\CurrentVersion\AppModel\SystemAppData\Microsoft.549981C3F5F10_8wekyb3d8bbwe\CortanaStartupId" -Force
|
|
}
|
|
New-ItemProperty -Path "Registry::HKEY_CLASSES_ROOT\Local Settings\Software\Microsoft\Windows\CurrentVersion\AppModel\SystemAppData\Microsoft.549981C3F5F10_8wekyb3d8bbwe\CortanaStartupId" -Name State -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
"Enable"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.549981C3F5F10)
|
|
{
|
|
if (-not (Test-Path -Path "Registry::HKEY_CLASSES_ROOT\Local Settings\Software\Microsoft\Windows\CurrentVersion\AppModel\SystemAppData\Microsoft.549981C3F5F10_8wekyb3d8bbwe\CortanaStartupId"))
|
|
{
|
|
New-Item -Path "Registry::HKEY_CLASSES_ROOT\Local Settings\Software\Microsoft\Windows\CurrentVersion\AppModel\SystemAppData\Microsoft.549981C3F5F10_8wekyb3d8bbwe\CortanaStartupId" -Force
|
|
}
|
|
New-ItemProperty -Path "Registry::HKEY_CLASSES_ROOT\Local Settings\Software\Microsoft\Windows\CurrentVersion\AppModel\SystemAppData\Microsoft.549981C3F5F10_8wekyb3d8bbwe\CortanaStartupId" -Name State -PropertyType DWord -Value 2 -Force
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Microsoft Teams autostarting
|
|
|
|
.PARAMETER Disable
|
|
Enable Teams autostarting
|
|
|
|
.PARAMETER Enable
|
|
Disable Teams autostarting
|
|
|
|
.EXAMPLE
|
|
TeamsAutostart -Disable
|
|
|
|
.EXAMPLE
|
|
TeamsAutostart -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function TeamsAutostart
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if (Get-AppxPackage -Name MicrosoftTeams)
|
|
{
|
|
if (-not (Test-Path -Path "HKCU:\Software\Classes\Local Settings\Software\Microsoft\Windows\CurrentVersion\AppModel\SystemAppData\MicrosoftTeams_8wekyb3d8bbwe\TeamsStartupTask"))
|
|
{
|
|
New-Item -Path "HKCU:\Software\Classes\Local Settings\Software\Microsoft\Windows\CurrentVersion\AppModel\SystemAppData\MicrosoftTeams_8wekyb3d8bbwe\TeamsStartupTask" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\Software\Classes\Local Settings\Software\Microsoft\Windows\CurrentVersion\AppModel\SystemAppData\MicrosoftTeams_8wekyb3d8bbwe\TeamsStartupTask" -Name State -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
"Enable"
|
|
{
|
|
if (Get-AppxPackage -Name MicrosoftTeams)
|
|
{
|
|
if (-not (Test-Path -Path "HKCU:\Software\Classes\Local Settings\Software\Microsoft\Windows\CurrentVersion\AppModel\SystemAppData\MicrosoftTeams_8wekyb3d8bbwe\TeamsStartupTask"))
|
|
{
|
|
New-Item -Path "HKCU:\Software\Classes\Local Settings\Software\Microsoft\Windows\CurrentVersion\AppModel\SystemAppData\MicrosoftTeams_8wekyb3d8bbwe\TeamsStartupTask" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\Software\Classes\Local Settings\Software\Microsoft\Windows\CurrentVersion\AppModel\SystemAppData\MicrosoftTeams_8wekyb3d8bbwe\TeamsStartupTask" -Name State -PropertyType DWord -Value 2 -Force
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# Check for UWP apps updates
|
|
function CheckUWPAppsUpdates
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
Get-CimInstance -Namespace "Root\cimv2\mdm\dmmap" -ClassName "MDM_EnterpriseModernAppManagement_AppManagement01" | Invoke-CimMethod -MethodName UpdateScanMethod
|
|
}
|
|
#endregion UWP apps
|
|
|
|
#region Gaming
|
|
<#
|
|
.SYNOPSIS
|
|
Xbox Game Bar
|
|
|
|
.PARAMETER Disable
|
|
Disable Xbox Game Bar
|
|
|
|
.PARAMETER Enable
|
|
Enable Xbox Game Bar
|
|
|
|
.EXAMPLE
|
|
XboxGameBar -Disable
|
|
|
|
.EXAMPLE
|
|
XboxGameBar -Enable
|
|
|
|
.NOTES
|
|
To prevent popping up the "You'll need a new app to open this ms-gamingoverlay" warning, you need to disable the Xbox Game Bar app, even if you uninstalled it before
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function XboxGameBar
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\GameDVR -Name AppCaptureEnabled -PropertyType DWord -Value 0 -Force
|
|
New-ItemProperty -Path HKCU:\System\GameConfigStore -Name GameDVR_Enabled -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\GameDVR -Name AppCaptureEnabled -PropertyType DWord -Value 1 -Force
|
|
New-ItemProperty -Path HKCU:\System\GameConfigStore -Name GameDVR_Enabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Xbox Game Bar tips
|
|
|
|
.PARAMETER Disable
|
|
Disable Xbox Game Bar tips
|
|
|
|
.PARAMETER Enable
|
|
Enable Xbox Game Bar tips
|
|
|
|
.EXAMPLE
|
|
XboxGameTips -Disable
|
|
|
|
.EXAMPLE
|
|
XboxGameTips -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function XboxGameTips
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if ((Get-AppxPackage -Name Microsoft.XboxGamingOverlay) -or (Get-AppxPackage -Name Microsoft.GamingApp))
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\GameBar -Name ShowStartupPanel -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
"Enable"
|
|
{
|
|
if ((Get-AppxPackage -Name Microsoft.XboxGamingOverlay) -or (Get-AppxPackage -Name Microsoft.GamingApp))
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\GameBar -Name ShowStartupPanel -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Choose an app and set the "High performance" graphics performance for it
|
|
|
|
.EXAMPLE
|
|
SetAppGraphicsPerformance
|
|
|
|
.NOTES
|
|
Works only with a dedicated GPU
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function SetAppGraphicsPerformance
|
|
{
|
|
if (Get-CimInstance -ClassName Win32_VideoController | Where-Object -FilterScript {($_.AdapterDACType -ne "Internal") -and ($null -ne $_.AdapterDACType)})
|
|
{
|
|
$Title = $Localization.GraphicsPerformanceTitle
|
|
$Message = $Localization.GraphicsPerformanceRequest
|
|
$Yes = $Localization.Yes
|
|
$No = $Localization.No
|
|
$Options = "&$Yes", "&$No"
|
|
$DefaultChoice = 1
|
|
|
|
do
|
|
{
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
Add-Type -AssemblyName System.Windows.Forms
|
|
$OpenFileDialog = New-Object -TypeName System.Windows.Forms.OpenFileDialog
|
|
$OpenFileDialog.Filter = $Localization.EXEFilesFilter
|
|
$OpenFileDialog.InitialDirectory = "::{20D04FE0-3AEA-1069-A2D8-08002B30309D}"
|
|
$OpenFileDialog.Multiselect = $false
|
|
|
|
# Force move the open file dialog to the foreground
|
|
$Focus = New-Object -TypeName System.Windows.Forms.Form -Property @{TopMost = $true}
|
|
$OpenFileDialog.ShowDialog($Focus)
|
|
|
|
if ($OpenFileDialog.FileName)
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\DirectX\UserGpuPreferences))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\DirectX\UserGpuPreferences -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\DirectX\UserGpuPreferences -Name $OpenFileDialog.FileName -PropertyType String -Value "GpuPreference=2;" -Force
|
|
Write-Verbose -Message ("{0}" -f $OpenFileDialog.FileName) -Verbose
|
|
}
|
|
}
|
|
"1"
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Skipped -Verbose
|
|
}
|
|
}
|
|
}
|
|
until ($Result -eq 1)
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Hardware-accelerated GPU scheduling
|
|
|
|
.PARAMETER Enable
|
|
Enable hardware-accelerated GPU scheduling
|
|
|
|
.PARAMETER Disable
|
|
Disable hardware-accelerated GPU scheduling
|
|
|
|
.EXAMPLE
|
|
GPUScheduling -Enable
|
|
|
|
.EXAMPLE
|
|
GPUScheduling -Disable
|
|
|
|
.NOTES
|
|
Only with a dedicated GPU and WDDM verion is 2.7 or higher. Restart needed
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function GPUScheduling
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
if (Get-CimInstance -ClassName CIM_VideoController | Where-Object -FilterScript {($_.AdapterDACType -ne "Internal") -and ($null -ne $_.AdapterDACType)})
|
|
{
|
|
# Determining whether an OS is not installed on a virtual machine
|
|
if ((Get-CimInstance -ClassName CIM_ComputerSystem).Model -notmatch "Virtual")
|
|
{
|
|
# Checking whether a WDDM verion is 2.7 or higher
|
|
$WddmVersion_Min = Get-ItemPropertyValue -Path HKLM:\SYSTEM\CurrentControlSet\Control\GraphicsDrivers\FeatureSetUsage -Name WddmVersion_Min
|
|
if ($WddmVersion_Min -ge 2700)
|
|
{
|
|
New-ItemProperty -Path "HKLM:\SYSTEM\CurrentControlSet\Control\GraphicsDrivers" -Name HwSchMode -PropertyType DWord -Value 2 -Force
|
|
}
|
|
}
|
|
}
|
|
}
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path "HKLM:\SYSTEM\CurrentControlSet\Control\GraphicsDrivers" -Name HwSchMode -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
#endregion Gaming
|
|
|
|
#region Scheduled tasks
|
|
<#
|
|
.SYNOPSIS
|
|
The "Windows Cleanup" scheduled task for cleaning up Windows unused files and updates
|
|
|
|
.PARAMETER Register
|
|
Create the "Windows Cleanup" scheduled task for cleaning up Windows unused files and updates
|
|
|
|
.PARAMETER Delete
|
|
Delete the "Windows Cleanup" and "Windows Cleanup Notification" scheduled tasks for cleaning up Windows unused files and updates
|
|
|
|
.EXAMPLE
|
|
CleanupTask -Register
|
|
|
|
.EXAMPLE
|
|
CleanupTask -Delete
|
|
|
|
.NOTES
|
|
A native interactive toast notification pops up every 30 days
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function CleanupTask
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Register"
|
|
)]
|
|
[switch]
|
|
$Register,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Delete"
|
|
)]
|
|
[switch]
|
|
$Delete
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Register"
|
|
{
|
|
if (-not (Get-ScheduledTask -TaskPath "\SophiApp\" -TaskName "Windows Cleanup" -ErrorAction Ignore))
|
|
{
|
|
Get-ChildItem -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\VolumeCaches | ForEach-Object -Process {
|
|
Remove-ItemProperty -Path $_.PsPath -Name StateFlags1337 -Force -ErrorAction Ignore
|
|
}
|
|
|
|
$VolumeCaches = @(
|
|
"Delivery Optimization Files",
|
|
"BranchCache",
|
|
"Device Driver Packages",
|
|
"Language Pack",
|
|
"Previous Installations",
|
|
"Setup Log Files",
|
|
"System error memory dump files",
|
|
"System error minidump files",
|
|
"Temporary Setup Files",
|
|
"Update Cleanup",
|
|
"Windows Defender",
|
|
"Windows ESD installation files",
|
|
"Windows Upgrade Log Files"
|
|
)
|
|
foreach ($VolumeCache in $VolumeCaches)
|
|
{
|
|
if (-not (Test-Path -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\VolumeCaches\$VolumeCache"))
|
|
{
|
|
New-Item -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\VolumeCaches\$VolumeCache" -Force
|
|
}
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\VolumeCaches\$VolumeCache" -Name StateFlags1337 -PropertyType DWord -Value 2 -Force
|
|
}
|
|
|
|
$CleanupTask = @"
|
|
Get-Process -Name cleanmgr | Stop-Process -Force
|
|
Get-Process -Name Dism | Stop-Process -Force
|
|
Get-Process -Name DismHost | Stop-Process -Force
|
|
|
|
`$ProcessInfo = New-Object -TypeName System.Diagnostics.ProcessStartInfo
|
|
`$ProcessInfo.FileName = """$env:SystemRoot\system32\cleanmgr.exe"""
|
|
`$ProcessInfo.Arguments = """/sagerun:1337"""
|
|
`$ProcessInfo.UseShellExecute = `$true
|
|
`$ProcessInfo.WindowStyle = [System.Diagnostics.ProcessWindowStyle]::Minimized
|
|
|
|
`$Process = New-Object -TypeName System.Diagnostics.Process
|
|
`$Process.StartInfo = `$ProcessInfo
|
|
`$Process.Start() | Out-Null
|
|
|
|
Start-Sleep -Seconds 3
|
|
|
|
[int]`$SourceMainWindowHandle = (Get-Process -Name cleanmgr | Where-Object -FilterScript {`$_.PriorityClass -eq """BelowNormal"""}).MainWindowHandle
|
|
|
|
function MinimizeWindow
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(Mandatory = `$true)]
|
|
`$Process
|
|
)
|
|
|
|
`$ShowWindowAsync = @{
|
|
Namespace = """WinAPI"""
|
|
Name = """Win32ShowWindowAsync"""
|
|
Language = """CSharp"""
|
|
MemberDefinition = @'
|
|
[DllImport("""user32.dll""")]
|
|
public static extern bool ShowWindowAsync(IntPtr hWnd, int nCmdShow);
|
|
'@
|
|
}
|
|
|
|
if (-not ("""WinAPI.Win32ShowWindowAsync""" -as [type]))
|
|
{
|
|
Add-Type @ShowWindowAsync
|
|
}
|
|
`$MainWindowHandle = (Get-Process -Name `$Process | Where-Object -FilterScript {`$_.PriorityClass -eq """BelowNormal"""}).MainWindowHandle
|
|
[WinAPI.Win32ShowWindowAsync]::ShowWindowAsync(`$MainWindowHandle, 2)
|
|
}
|
|
|
|
while (`$true)
|
|
{
|
|
[int]`$CurrentMainWindowHandle = (Get-Process -Name cleanmgr | Where-Object -FilterScript {`$_.PriorityClass -eq """BelowNormal"""}).MainWindowHandle
|
|
if (`$SourceMainWindowHandle -ne `$CurrentMainWindowHandle)
|
|
{
|
|
MinimizeWindow -Process cleanmgr
|
|
break
|
|
}
|
|
Start-Sleep -Milliseconds 5
|
|
}
|
|
|
|
`$ProcessInfo = New-Object -TypeName System.Diagnostics.ProcessStartInfo
|
|
`$ProcessInfo.FileName = """`$env:SystemRoot\system32\dism.exe"""
|
|
`$ProcessInfo.Arguments = """/Online /English /Cleanup-Image /StartComponentCleanup /NoRestart"""
|
|
`$ProcessInfo.UseShellExecute = `$true
|
|
`$ProcessInfo.WindowStyle = [System.Diagnostics.ProcessWindowStyle]::Minimized
|
|
|
|
`$Process = New-Object -TypeName System.Diagnostics.Process
|
|
`$Process.StartInfo = `$ProcessInfo
|
|
`$Process.Start() | Out-Null
|
|
"@
|
|
|
|
# Create the "Windows Cleanup" task
|
|
$Action = New-ScheduledTaskAction -Execute powershell.exe -Argument "-WindowStyle Hidden -Command $CleanupTask"
|
|
$Settings = New-ScheduledTaskSettingsSet -Compatibility Win8 -StartWhenAvailable
|
|
$Principal = New-ScheduledTaskPrincipal -UserId $env:USERNAME -RunLevel Highest
|
|
$Parameters = @{
|
|
TaskName = "Windows Cleanup"
|
|
TaskPath = "Sophia Script"
|
|
Principal = $Principal
|
|
Action = $Action
|
|
Description = $Localization.CleanupTaskDescription
|
|
Settings = $Settings
|
|
}
|
|
Register-ScheduledTask @Parameters -Force
|
|
|
|
# Persist the Settings notifications to prevent to immediately disappear from Action Center
|
|
if (-not (Test-Path -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Notifications\Settings\windows.immersivecontrolpanel_cw5n1h2txyewy!microsoft.windows.immersivecontrolpanel"))
|
|
{
|
|
New-Item -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Notifications\Settings\windows.immersivecontrolpanel_cw5n1h2txyewy!microsoft.windows.immersivecontrolpanel" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Notifications\Settings\windows.immersivecontrolpanel_cw5n1h2txyewy!microsoft.windows.immersivecontrolpanel" -Name ShowInActionCenter -PropertyType DWord -Value 1 -Force
|
|
|
|
# Register the "WindowsCleanup" protocol to be able to run the scheduled task by clicking the "Run" button in a toast
|
|
if (-not (Test-Path -Path Registry::HKEY_CLASSES_ROOT\WindowsCleanup\shell\open\command))
|
|
{
|
|
New-Item -Path Registry::HKEY_CLASSES_ROOT\WindowsCleanup\shell\open\command -Force
|
|
}
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\WindowsCleanup -Name "(default)" -PropertyType String -Value "URL:WindowsCleanup" -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\WindowsCleanup -Name "URL Protocol" -PropertyType String -Value "" -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\WindowsCleanup -Name EditFlags -PropertyType DWord -Value 2162688 -Force
|
|
|
|
# Start the "Windows Cleanup" task if the "Run" button clicked
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\WindowsCleanup\shell\open\command -Name "(default)" -PropertyType String -Value 'powershell.exe -Command "& {Start-ScheduledTask -TaskPath ''\Sophia Script\'' -TaskName ''Windows Cleanup''}"' -Force
|
|
|
|
$ToastNotification = @"
|
|
[Windows.UI.Notifications.ToastNotificationManager, Windows.UI.Notifications, ContentType = WindowsRuntime] | Out-Null
|
|
[Windows.Data.Xml.Dom.XmlDocument, Windows.Data.Xml.Dom.XmlDocument, ContentType = WindowsRuntime] | Out-Null
|
|
|
|
[xml]`$ToastTemplate = @"""
|
|
<toast duration="""Long""" scenario="""reminder""">
|
|
<visual>
|
|
<binding template="""ToastGeneric""">
|
|
<text>$($Localization.CleanupTaskNotificationTitle)</text>
|
|
<group>
|
|
<subgroup>
|
|
<text hint-style="""title""" hint-wrap="""true""">$($Localization.CleanupTaskNotificationEventTitle)</text>
|
|
</subgroup>
|
|
</group>
|
|
<group>
|
|
<subgroup>
|
|
<text hint-style="""body""" hint-wrap="""true""">$($Localization.CleanupTaskNotificationEvent)</text>
|
|
</subgroup>
|
|
</group>
|
|
</binding>
|
|
</visual>
|
|
<audio src="""ms-winsoundevent:notification.default""" />
|
|
<actions>
|
|
<input id="""SnoozeTimer""" type="""selection""" title="""$($Localization.CleanupTaskNotificationSnoozeInterval)""" defaultInput="""1""">
|
|
<selection id="""1""" content="""$($Localization.Minute)""" />
|
|
<selection id="""30""" content="""$($Localization.HalfHour)""" />
|
|
<selection id="""240""" content="""$($Localization.FourHours)""" />
|
|
</input>
|
|
<action activationType="""system""" arguments="""snooze""" hint-inputId="""SnoozeTimer""" content="""""" id="""test-snooze"""/>
|
|
<action arguments="""WindowsCleanup:""" content="""$($Localization.Run)""" activationType="""protocol"""/>
|
|
<action arguments="""dismiss""" content="""""" activationType="""system"""/>
|
|
</actions>
|
|
</toast>
|
|
"""@
|
|
|
|
`$ToastXml = [Windows.Data.Xml.Dom.XmlDocument]::New()
|
|
`$ToastXml.LoadXml(`$ToastTemplate.OuterXml)
|
|
|
|
`$ToastMessage = [Windows.UI.Notifications.ToastNotification]::New(`$ToastXML)
|
|
[Windows.UI.Notifications.ToastNotificationManager]::CreateToastNotifier("""windows.immersivecontrolpanel_cw5n1h2txyewy!microsoft.windows.immersivecontrolpanel""").Show(`$ToastMessage)
|
|
"@
|
|
|
|
# Create the "Windows Cleanup Notification" task
|
|
$Action = New-ScheduledTaskAction -Execute powershell.exe -Argument "-WindowStyle Hidden -Command $ToastNotification"
|
|
$Settings = New-ScheduledTaskSettingsSet -Compatibility Win8 -StartWhenAvailable
|
|
$Principal = New-ScheduledTaskPrincipal -UserId $env:USERNAME -RunLevel Highest
|
|
$Trigger = New-ScheduledTaskTrigger -Daily -DaysInterval 30 -At 9pm
|
|
$Parameters = @{
|
|
TaskName = "Windows Cleanup Notification"
|
|
TaskPath = "Sophia Script"
|
|
Action = $Action
|
|
Settings = $Settings
|
|
Principal = $Principal
|
|
Trigger = $Trigger
|
|
Description = $Localization.CleanupNotificationTaskDescription
|
|
}
|
|
Register-ScheduledTask @Parameters -Force
|
|
}
|
|
}
|
|
"Delete"
|
|
{
|
|
Get-ChildItem -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\VolumeCaches | ForEach-Object -Process {
|
|
Remove-ItemProperty -Path $_.PsPath -Name StateFlags1337 -Force -ErrorAction Ignore
|
|
}
|
|
|
|
Remove-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Notifications\Settings\windows.immersivecontrolpanel_cw5n1h2txyewy!microsoft.windows.immersivecontrolpanel" -Name ShowInActionCenter -Force -ErrorAction Ignore
|
|
|
|
Unregister-ScheduledTask -TaskPath "\Sophia Script\" -TaskName "Windows Cleanup", "Windows Cleanup Notification" -Confirm:$false -ErrorAction Ignore
|
|
|
|
Remove-Item -Path Registry::HKEY_CLASSES_ROOT\WindowsCleanup -Recurse -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "SoftwareDistribution" scheduled task for cleaning up the %SystemRoot%\SoftwareDistribution\Download folder
|
|
|
|
.PARAMETER Register
|
|
Create the "SoftwareDistribution" scheduled task for cleaning up the %SystemRoot%\SoftwareDistribution\Download folder
|
|
|
|
.PARAMETER Delete
|
|
Delete the "SoftwareDistribution" scheduled task for cleaning up the %SystemRoot%\SoftwareDistribution\Download folder
|
|
|
|
.EXAMPLE
|
|
SoftwareDistributionTask -Register
|
|
|
|
.EXAMPLE
|
|
SoftwareDistributionTask -Delete
|
|
|
|
.NOTES
|
|
The task will wait until the Windows Updates service finishes running. The task runs every 90 days
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function SoftwareDistributionTask
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Register"
|
|
)]
|
|
[switch]
|
|
$Register,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Delete"
|
|
)]
|
|
[switch]
|
|
$Delete
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Register"
|
|
{
|
|
if (-not (Get-ScheduledTask -TaskPath "\SophiApp\" -TaskName "SoftwareDistribution" -ErrorAction Ignore))
|
|
{
|
|
# Persist the Settings notifications to prevent to immediately disappear from Action Center
|
|
if (-not (Test-Path -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Notifications\Settings\windows.immersivecontrolpanel_cw5n1h2txyewy!microsoft.windows.immersivecontrolpanel"))
|
|
{
|
|
New-Item -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Notifications\Settings\windows.immersivecontrolpanel_cw5n1h2txyewy!microsoft.windows.immersivecontrolpanel" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Notifications\Settings\windows.immersivecontrolpanel_cw5n1h2txyewy!microsoft.windows.immersivecontrolpanel" -Name ShowInActionCenter -PropertyType DWord -Value 1 -Force
|
|
|
|
$SoftwareDistributionTask = @"
|
|
(Get-Service -Name wuauserv).WaitForStatus('Stopped', '01:00:00')
|
|
Get-ChildItem -Path `$env:SystemRoot\SoftwareDistribution\Download -Recurse -Force | Remove-Item -Recurse -Force
|
|
|
|
[Windows.UI.Notifications.ToastNotificationManager, Windows.UI.Notifications, ContentType = WindowsRuntime] | Out-Null
|
|
[Windows.Data.Xml.Dom.XmlDocument, Windows.Data.Xml.Dom.XmlDocument, ContentType = WindowsRuntime] | Out-Null
|
|
|
|
[xml]`$ToastTemplate = @"""
|
|
<toast duration="""Long""">
|
|
<visual>
|
|
<binding template="""ToastGeneric""">
|
|
<text>$($Localization.TaskNotificationTitle)</text>
|
|
<group>
|
|
<subgroup>
|
|
<text hint-style="""body""" hint-wrap="""true""">$($Localization.SoftwareDistributionTaskNotificationEvent)</text>
|
|
</subgroup>
|
|
</group>
|
|
</binding>
|
|
</visual>
|
|
<audio src="""ms-winsoundevent:notification.default""" />
|
|
</toast>
|
|
"""@
|
|
|
|
`$ToastXml = [Windows.Data.Xml.Dom.XmlDocument]::New()
|
|
`$ToastXml.LoadXml(`$ToastTemplate.OuterXml)
|
|
|
|
`$ToastMessage = [Windows.UI.Notifications.ToastNotification]::New(`$ToastXML)
|
|
[Windows.UI.Notifications.ToastNotificationManager]::CreateToastNotifier("""windows.immersivecontrolpanel_cw5n1h2txyewy!microsoft.windows.immersivecontrolpanel""").Show(`$ToastMessage)
|
|
"@
|
|
|
|
# Create the "SoftwareDistribution" task
|
|
$Action = New-ScheduledTaskAction -Execute powershell.exe -Argument "-WindowStyle Hidden -Command $SoftwareDistributionTask"
|
|
$Settings = New-ScheduledTaskSettingsSet -Compatibility Win8 -StartWhenAvailable
|
|
$Principal = New-ScheduledTaskPrincipal -UserId $env:USERNAME -RunLevel Highest
|
|
$Trigger = New-ScheduledTaskTrigger -Daily -DaysInterval 90 -At 9pm
|
|
$Parameters = @{
|
|
TaskName = "SoftwareDistribution"
|
|
TaskPath = "Sophia Script"
|
|
Action = $Action
|
|
Settings = $Settings
|
|
Principal = $Principal
|
|
Trigger = $Trigger
|
|
Description = $Localization.FolderTaskDescription -f "%SystemRoot%\SoftwareDistribution\Download"
|
|
}
|
|
Register-ScheduledTask @Parameters -Force
|
|
}
|
|
}
|
|
"Delete"
|
|
{
|
|
Unregister-ScheduledTask -TaskPath "\Sophia Script\" -TaskName SoftwareDistribution -Confirm:$false -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Temp" scheduled task for cleaning up the %TEMP% folder
|
|
|
|
.PARAMETER Register
|
|
Create the "Temp" scheduled task for cleaning up the %TEMP% folder
|
|
|
|
.PARAMETER Delete
|
|
Delete the "Temp" scheduled task for cleaning up the %TEMP% folder
|
|
|
|
.EXAMPLE
|
|
TempTask -Register
|
|
|
|
.EXAMPLE
|
|
TempTask -Delete
|
|
|
|
.NOTES
|
|
Only files older than one day will be deleted. The task runs every 60 days
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function TempTask
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Register"
|
|
)]
|
|
[switch]
|
|
$Register,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Delete"
|
|
)]
|
|
[switch]
|
|
$Delete
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Register"
|
|
{
|
|
if (-not (Get-ScheduledTask -TaskPath "\SophiApp\" -TaskName "Temp" -ErrorAction Ignore))
|
|
{
|
|
$TempTask = @"
|
|
Get-ChildItem -Path `$env:TEMP -Recurse -Force | Where-Object -FilterScript {`$_.CreationTime -lt (Get-Date).AddDays(-1)} | Remove-Item -Recurse -Force
|
|
|
|
[Windows.UI.Notifications.ToastNotificationManager, Windows.UI.Notifications, ContentType = WindowsRuntime] | Out-Null
|
|
[Windows.Data.Xml.Dom.XmlDocument, Windows.Data.Xml.Dom.XmlDocument, ContentType = WindowsRuntime] | Out-Null
|
|
|
|
[xml]`$ToastTemplate = @"""
|
|
<toast duration="""Long""">
|
|
<visual>
|
|
<binding template="""ToastGeneric""">
|
|
<text>$($Localization.TaskNotificationTitle)</text>
|
|
<group>
|
|
<subgroup>
|
|
<text hint-style="""body""" hint-wrap="""true""">$($Localization.TempTaskNotificationEvent)</text>
|
|
</subgroup>
|
|
</group>
|
|
</binding>
|
|
</visual>
|
|
<audio src="""ms-winsoundevent:notification.default""" />
|
|
</toast>
|
|
"""@
|
|
|
|
`$ToastXml = [Windows.Data.Xml.Dom.XmlDocument]::New()
|
|
`$ToastXml.LoadXml(`$ToastTemplate.OuterXml)
|
|
|
|
`$ToastMessage = [Windows.UI.Notifications.ToastNotification]::New(`$ToastXML)
|
|
[Windows.UI.Notifications.ToastNotificationManager]::CreateToastNotifier("""windows.immersivecontrolpanel_cw5n1h2txyewy!microsoft.windows.immersivecontrolpanel""").Show(`$ToastMessage)
|
|
"@
|
|
|
|
# Create the "Temp" task
|
|
$Action = New-ScheduledTaskAction -Execute powershell.exe -Argument "-WindowStyle Hidden -Command $TempTask"
|
|
$Settings = New-ScheduledTaskSettingsSet -Compatibility Win8 -StartWhenAvailable
|
|
$Principal = New-ScheduledTaskPrincipal -UserId $env:USERNAME -RunLevel Highest
|
|
$Trigger = New-ScheduledTaskTrigger -Daily -DaysInterval 60 -At 9pm
|
|
$Parameters = @{
|
|
TaskName = "Temp"
|
|
TaskPath = "Sophia Script"
|
|
Action = $Action
|
|
Settings = $Settings
|
|
Principal = $Principal
|
|
Trigger = $Trigger
|
|
Description = $Localization.FolderTaskDescription -f "%TEMP%"
|
|
}
|
|
Register-ScheduledTask @Parameters -Force
|
|
}
|
|
}
|
|
"Delete"
|
|
{
|
|
Unregister-ScheduledTask -TaskPath "\Sophia Script\" -TaskName Temp -Confirm:$false -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
#endregion Scheduled tasks
|
|
|
|
#region Microsoft Defender & Security
|
|
<#
|
|
.SYNOPSIS
|
|
Microsoft Defender Exploit Guard network protection
|
|
|
|
.PARAMETER Enable
|
|
Enable Microsoft Defender Exploit Guard network protection
|
|
|
|
.PARAMETER Disable
|
|
Disable Microsoft Defender Exploit Guard network protection
|
|
|
|
.EXAMPLE
|
|
NetworkProtection -Enable
|
|
|
|
.EXAMPLE
|
|
NetworkProtection -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function NetworkProtection
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
if ($Script:DefenderEnabled)
|
|
{
|
|
Set-MpPreference -EnableNetworkProtection Enabled
|
|
}
|
|
}
|
|
"Disable"
|
|
{
|
|
if ($Script:DefenderEnabled)
|
|
{
|
|
Set-MpPreference -EnableNetworkProtection Disabled
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Detection for potentially unwanted applications
|
|
|
|
.PARAMETER Enable
|
|
Enable detection for potentially unwanted applications and block them
|
|
|
|
.PARAMETER Disable
|
|
Disable detection for potentially unwanted applications and block them
|
|
|
|
.EXAMPLE
|
|
PUAppsDetection -Enable
|
|
|
|
.EXAMPLE
|
|
PUAppsDetection -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function PUAppsDetection
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
if ($Script:DefenderEnabled)
|
|
{
|
|
Set-MpPreference -PUAProtection Enabled
|
|
}
|
|
}
|
|
"Disable"
|
|
{
|
|
if ($Script:DefenderEnabled)
|
|
{
|
|
Set-MpPreference -PUAProtection Disabled
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# Dismiss Microsoft Defender offer in the Windows Security about signing in Microsoft account
|
|
function DismissMSAccount
|
|
{
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows Security Health\State" -Name AccountProtection_MicrosoftAccount_Disconnected -PropertyType DWord -Value 1 -Force
|
|
}
|
|
|
|
# Dismiss Microsoft Defender offer in the Windows Security about turning on the SmartScreen filter for Microsoft Edge
|
|
function DismissSmartScreenFilter
|
|
{
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows Security Health\State" -Name AppAndBrowser_EdgeSmartScreenOff -PropertyType DWord -Value 0 -Force
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Audit process creation
|
|
|
|
.PARAMETER Enable
|
|
Enable events auditing generated when a process is created (starts)
|
|
|
|
.PARAMETER Disable
|
|
Disable events auditing generated when a process is created (starts)
|
|
|
|
.EXAMPLE
|
|
AuditProcess -Enable
|
|
|
|
.EXAMPLE
|
|
AuditProcess -Disable
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function AuditProcess
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
auditpol /set /subcategory:"{0CCE922B-69AE-11D9-BED3-505054503030}" /success:enable /failure:enable
|
|
}
|
|
"Disable"
|
|
{
|
|
auditpol /set /subcategory:"{0CCE922B-69AE-11D9-BED3-505054503030}" /success:disable /failure:disable
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Сommand line auditing
|
|
|
|
.PARAMETER Enable
|
|
Include command line in process creation events
|
|
|
|
.PARAMETER Disable
|
|
Do not include command line in process creation events
|
|
|
|
.EXAMPLE
|
|
CommandLineProcessAudit -Enable
|
|
|
|
.EXAMPLE
|
|
CommandLineProcessAudit -Disable
|
|
|
|
.NOTES
|
|
In order this feature to work events auditing (ProcessAudit -Enable) will be enabled
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function CommandLineProcessAudit
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
# Enable events auditing generated when a process is created (starts)
|
|
auditpol /set /subcategory:"{0CCE922B-69AE-11D9-BED3-505054503030}" /success:enable /failure:enable
|
|
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\System\Audit -Name ProcessCreationIncludeCmdLine_Enabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
Remove-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\System\Audit -Name ProcessCreationIncludeCmdLine_Enabled -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Process Creation" Event Viewer custom view
|
|
|
|
.PARAMETER Enable
|
|
Create the "Process Creation" сustom view in the Event Viewer to log executed processes and their arguments
|
|
|
|
.PARAMETER Disable
|
|
Remove the "Process Creation" custom view in the Event Viewer
|
|
|
|
.EXAMPLE
|
|
EventViewerCustomView -Enable
|
|
|
|
.EXAMPLE
|
|
EventViewerCustomView -Disable
|
|
|
|
.NOTES
|
|
In order this feature to work events auditing (ProcessAudit -Enable) and command line (CommandLineProcessAudit -Enable) in process creation events will be enabled
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function EventViewerCustomView
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
# Enable events auditing generated when a process is created (starts)
|
|
auditpol /set /subcategory:"{0CCE922B-69AE-11D9-BED3-505054503030}" /success:enable /failure:enable
|
|
|
|
# Include command line in process creation events
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\System\Audit -Name ProcessCreationIncludeCmdLine_Enabled -PropertyType DWord -Value 1 -Force
|
|
|
|
$XML = @"
|
|
<ViewerConfig>
|
|
<QueryConfig>
|
|
<QueryParams>
|
|
<UserQuery />
|
|
</QueryParams>
|
|
<QueryNode>
|
|
<Name>$($Localization.EventViewerCustomViewName)</Name>
|
|
<Description>$($Localization.EventViewerCustomViewDescription)</Description>
|
|
<QueryList>
|
|
<Query Id="0" Path="Security">
|
|
<Select Path="Security">*[System[(EventID=4688)]]</Select>
|
|
</Query>
|
|
</QueryList>
|
|
</QueryNode>
|
|
</QueryConfig>
|
|
</ViewerConfig>
|
|
"@
|
|
|
|
if (-not (Test-Path -Path "$env:ProgramData\Microsoft\Event Viewer\Views"))
|
|
{
|
|
New-Item -Path "$env:ProgramData\Microsoft\Event Viewer\Views" -ItemType Directory -Force
|
|
}
|
|
|
|
# Save ProcessCreation.xml in the UTF-8 with BOM encoding
|
|
Set-Content -Path "$env:ProgramData\Microsoft\Event Viewer\Views\ProcessCreation.xml" -Value $XML -Encoding UTF8 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
Remove-Item -Path "$env:ProgramData\Microsoft\Event Viewer\Views\ProcessCreation.xml" -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Logging for all Windows PowerShell modules
|
|
|
|
.PARAMETER Enable
|
|
Enable logging for all Windows PowerShell modules
|
|
|
|
.PARAMETER Disable
|
|
Disable logging for all Windows PowerShell modules
|
|
|
|
.EXAMPLE
|
|
PowerShellModulesLogging -Enable
|
|
|
|
.EXAMPLE
|
|
PowerShellModulesLogging -Disable
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function PowerShellModulesLogging
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
if (-not (Test-Path -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\PowerShell\ModuleLogging\ModuleNames))
|
|
{
|
|
New-Item -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\PowerShell\ModuleLogging\ModuleNames -Force
|
|
}
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\PowerShell\ModuleLogging -Name EnableModuleLogging -PropertyType DWord -Value 1 -Force
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\PowerShell\ModuleLogging\ModuleNames -Name * -PropertyType String -Value * -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
Remove-ItemProperty -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\PowerShell\ModuleLogging -Name EnableModuleLogging -Force -ErrorAction Ignore
|
|
Remove-ItemProperty -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\PowerShell\ModuleLogging\ModuleNames -Name * -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Logging for all PowerShell scripts input to the Windows PowerShell event log
|
|
|
|
.PARAMETER Enable
|
|
Enable logging for all PowerShell scripts input to the Windows PowerShell event log
|
|
|
|
.PARAMETER Disable
|
|
Disable logging for all PowerShell scripts input to the Windows PowerShell event log
|
|
|
|
.EXAMPLE
|
|
PowerShellScriptsLogging -Enable
|
|
|
|
.EXAMPLE
|
|
PowerShellScriptsLogging -Disable
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function PowerShellScriptsLogging
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
if (-not (Test-Path -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\PowerShell\ScriptBlockLogging))
|
|
{
|
|
New-Item -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\PowerShell\ScriptBlockLogging -Force
|
|
}
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\PowerShell\ScriptBlockLogging -Name EnableScriptBlockLogging -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
Remove-ItemProperty -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\PowerShell\ScriptBlockLogging -Name EnableScriptBlockLogging -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Microsoft Defender SmartScreen
|
|
|
|
.PARAMETER Disable
|
|
Disable apps and files checking within Microsoft Defender SmartScreen
|
|
|
|
.PARAMETER Enable
|
|
Enable apps and files checking within Microsoft Defender SmartScreen
|
|
|
|
.EXAMPLE
|
|
AppsSmartScreen -Disable
|
|
|
|
.EXAMPLE
|
|
AppsSmartScreen -Enable
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function AppsSmartScreen
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer -Name SmartScreenEnabled -PropertyType String -Value Off -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer -Name SmartScreenEnabled -PropertyType String -Value Warn -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The Attachment Manager
|
|
|
|
.PARAMETER Disable
|
|
Microsoft Defender SmartScreen doesn't marks downloaded files from the Internet as unsafe
|
|
|
|
.PARAMETER Enable
|
|
Microsoft Defender SmartScreen marks downloaded files from the Internet as unsafe
|
|
|
|
.EXAMPLE
|
|
SaveZoneInformation -Disable
|
|
|
|
.EXAMPLE
|
|
SaveZoneInformation -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function SaveZoneInformation
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\Attachments))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\Attachments -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\Attachments -Name SaveZoneInformation -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
Remove-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\Attachments -Name SaveZoneInformation -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Windows Script Host
|
|
|
|
.PARAMETER Disable
|
|
Disable Windows Script Host
|
|
|
|
.PARAMETER Enable
|
|
Enable Windows Script Host
|
|
|
|
.EXAMPLE
|
|
WindowsScriptHost -Disable
|
|
|
|
.EXAMPLE
|
|
WindowsScriptHost -Enable
|
|
|
|
.NOTES
|
|
Blocks WSH from executing .js and .vbs files
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function WindowsScriptHost
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if (-not (Test-Path -Path "HKCU:\SOFTWARE\Microsoft\Windows Script Host\Settings"))
|
|
{
|
|
New-Item -Path "HKCU:\SOFTWARE\Microsoft\Windows Script Host\Settings" -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows Script Host\Settings" -Name Enabled -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
Remove-ItemProperty -Path "HKCU:\SOFTWARE\Microsoft\Windows Script Host\Settings" -Name Enabled -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Windows Sandbox
|
|
|
|
.PARAMETER Disable
|
|
Disable Windows Sandbox
|
|
|
|
.PARAMETER Enable
|
|
Enable Windows Sandbox
|
|
|
|
.EXAMPLE
|
|
WindowsSandbox -Disable
|
|
|
|
.EXAMPLE
|
|
WindowsSandbox -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function WindowsSandbox
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
if (Get-WindowsEdition -Online | Where-Object -FilterScript {($_.Edition -eq "Professional") -or ($_.Edition -like "Enterprise*")})
|
|
{
|
|
# Checking whether x86 virtualization is enabled in the firmware
|
|
if ((Get-CimInstance -ClassName CIM_Processor).VirtualizationFirmwareEnabled)
|
|
{
|
|
Disable-WindowsOptionalFeature -FeatureName Containers-DisposableClientVM -Online -NoRestart
|
|
}
|
|
else
|
|
{
|
|
try
|
|
{
|
|
# Determining whether Hyper-V is enabled
|
|
if ((Get-CimInstance -ClassName CIM_ComputerSystem).HypervisorPresent)
|
|
{
|
|
Disable-WindowsOptionalFeature -FeatureName Containers-DisposableClientVM -Online -NoRestart
|
|
}
|
|
}
|
|
catch [System.Exception]
|
|
{
|
|
Write-Error -Message $Localization.EnableHardwareVT -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
}
|
|
}
|
|
"Enable"
|
|
{
|
|
if (Get-WindowsEdition -Online | Where-Object -FilterScript {($_.Edition -eq "Professional") -or ($_.Edition -like "Enterprise*")})
|
|
{
|
|
# Checking whether x86 virtualization is enabled in the firmware
|
|
if ((Get-CimInstance -ClassName CIM_Processor).VirtualizationFirmwareEnabled)
|
|
{
|
|
Enable-WindowsOptionalFeature -FeatureName Containers-DisposableClientVM -All -Online -NoRestart
|
|
}
|
|
else
|
|
{
|
|
try
|
|
{
|
|
# Determining whether Hyper-V is enabled
|
|
if ((Get-CimInstance -ClassName CIM_ComputerSystem).HypervisorPresent)
|
|
{
|
|
Enable-WindowsOptionalFeature -FeatureName Containers-DisposableClientVM -All -Online -NoRestart
|
|
}
|
|
}
|
|
catch [System.Exception]
|
|
{
|
|
Write-Error -Message $Localization.EnableHardwareVT -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
DNS-over-HTTPS for IPv4
|
|
|
|
.PARAMETER Enable
|
|
Enable DNS-over-HTTPS for IPv4
|
|
|
|
.PARAMETER Disable
|
|
Disable DNS-over-HTTPS for IPv4
|
|
|
|
.EXAMPLE
|
|
DNSoverHTTPS -Enable -PrimaryDNS 1.0.0.1 -SecondaryDNS 1.1.1.1
|
|
|
|
.EXAMPLE
|
|
DNSoverHTTPS -Disable
|
|
|
|
.NOTES
|
|
The valid IPv4 addresses: 1.0.0.1, 1.1.1.1, 149.112.112.112, 8.8.4.4, 8.8.8.8, 9.9.9.9
|
|
|
|
.LINK
|
|
https://docs.microsoft.com/en-us/windows-server/networking/dns/doh-client-support
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function DNSoverHTTPS
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(Mandatory = $false)]
|
|
[ValidateSet("1.0.0.1", "1.1.1.1", "149.112.112.112", "8.8.4.4", "8.8.8.8", "9.9.9.9")]
|
|
# Isolate the IPv4 addresses only
|
|
[ValidateScript({
|
|
(@((Get-ChildItem -Path HKLM:\SYSTEM\CurrentControlSet\Services\Dnscache\Parameters\DohWellKnownServers).PSChildName) | Where-Object {$_ -notmatch ":"}) -contains $_
|
|
})]
|
|
[string]
|
|
$PrimaryDNS,
|
|
|
|
[Parameter(Mandatory = $false)]
|
|
[ValidateSet("1.0.0.1", "1.1.1.1", "149.112.112.112", "8.8.4.4", "8.8.8.8", "9.9.9.9")]
|
|
# Isolate the IPv4 addresses only
|
|
[ValidateScript({(@((Get-ChildItem -Path HKLM:\SYSTEM\CurrentControlSet\Services\Dnscache\Parameters\DohWellKnownServers).PSChildName) | Where-Object {$_ -notmatch ":"}) -contains $_})]
|
|
[string]
|
|
$SecondaryDNS,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
# Determining whether Hyper-V is enabled
|
|
# After enabling Hyper-V feature a virtual switch breing created, so we need to use different method to isolate the proper adapter
|
|
if (-not (Get-CimInstance -ClassName CIM_ComputerSystem).HypervisorPresent)
|
|
{
|
|
$InterfaceGuids = @((Get-NetAdapter -Physical).InterfaceGuid)
|
|
}
|
|
else
|
|
{
|
|
$InterfaceGuids = @((Get-NetRoute -AddressFamily IPv4 | Where-Object -FilterScript {$_.DestinationPrefix -eq "0.0.0.0/0"} | Get-NetAdapter).InterfaceGuid)
|
|
}
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
if (-not (Get-CimInstance -ClassName CIM_ComputerSystem).PartOfDomain)
|
|
{
|
|
# Set a primary and secondary DNS servers
|
|
if (-not (Get-CimInstance -ClassName CIM_ComputerSystem).HypervisorPresent)
|
|
{
|
|
Get-NetAdapter -Physical | Get-NetIPInterface -AddressFamily IPv4 | Set-DnsClientServerAddress -ServerAddresses $PrimaryDNS, $SecondaryDNS
|
|
}
|
|
else
|
|
{
|
|
Get-NetRoute | Where-Object -FilterScript {$_.DestinationPrefix -eq "0.0.0.0/0"} | Get-NetAdapter | Set-DnsClientServerAddress -ServerAddresses $PrimaryDNS, $SecondaryDNS
|
|
}
|
|
|
|
foreach ($InterfaceGuid in $InterfaceGuids)
|
|
{
|
|
if (-not (Test-Path -Path "HKLM:\SYSTEM\ControlSet001\Services\Dnscache\InterfaceSpecificParameters\$InterfaceGuid\DohInterfaceSettings\Doh\$PrimaryDNS"))
|
|
{
|
|
New-Item -Path "HKLM:\SYSTEM\ControlSet001\Services\Dnscache\InterfaceSpecificParameters\$InterfaceGuid\DohInterfaceSettings\Doh\$PrimaryDNS" -Force
|
|
}
|
|
if (-not (Test-Path -Path "HKLM:\SYSTEM\ControlSet001\Services\Dnscache\InterfaceSpecificParameters\$InterfaceGuid\DohInterfaceSettings\Doh\$SecondaryDNS"))
|
|
{
|
|
New-Item -Path "HKLM:\SYSTEM\ControlSet001\Services\Dnscache\InterfaceSpecificParameters\$InterfaceGuid\DohInterfaceSettings\Doh\$SecondaryDNS" -Force
|
|
}
|
|
# Encrypted preffered, unencrypted allowed
|
|
New-ItemProperty -Path "HKLM:\SYSTEM\ControlSet001\Services\Dnscache\InterfaceSpecificParameters\$InterfaceGuid\DohInterfaceSettings\Doh\$PrimaryDNS" -Name DohFlags -PropertyType QWord -Value 5 -Force
|
|
New-ItemProperty -Path "HKLM:\SYSTEM\ControlSet001\Services\Dnscache\InterfaceSpecificParameters\$InterfaceGuid\DohInterfaceSettings\Doh\$SecondaryDNS" -Name DohFlags -PropertyType QWord -Value 5 -Force
|
|
}
|
|
}
|
|
}
|
|
"Disable"
|
|
{
|
|
if (-not (Get-CimInstance -ClassName CIM_ComputerSystem).PartOfDomain)
|
|
{
|
|
# Determining whether Hyper-V is enabled
|
|
if (-not (Get-CimInstance -ClassName CIM_ComputerSystem).HypervisorPresent)
|
|
{
|
|
# Configure DNS servers automatically
|
|
Get-NetAdapter -Physical | Get-NetIPInterface -AddressFamily IPv4 | Set-DnsClientServerAddress -ResetServerAddresses
|
|
}
|
|
else
|
|
{
|
|
# Configure DNS servers automatically
|
|
Get-NetRoute | Where-Object -FilterScript {$_.DestinationPrefix -eq "0.0.0.0/0"} | Get-NetAdapter | Set-DnsClientServerAddress -ResetServerAddresses
|
|
}
|
|
|
|
foreach ($InterfaceGuid in $InterfaceGuids)
|
|
{
|
|
Remove-Item -Path "HKLM:\SYSTEM\ControlSet001\Services\Dnscache\InterfaceSpecificParameters\$InterfaceGuid\DohInterfaceSettings\Doh" -Recurse -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Clear-DnsClientCache
|
|
Register-DnsClient
|
|
}
|
|
#endregion Microsoft Defender & Security
|
|
|
|
#region Context menu
|
|
<#
|
|
.SYNOPSIS
|
|
The "Extract all" item in the Windows Installer (.msi) context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Extract all" item in the Windows Installer (.msi) context menu
|
|
|
|
.PARAMETER Remove
|
|
Hide the "Extract all" item from the Windows Installer (.msi) context menu
|
|
|
|
.EXAMPLE
|
|
MSIExtractContext -Show
|
|
|
|
.EXAMPLE
|
|
MSIExtractContext -Hide
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function MSIExtractContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Show"
|
|
{
|
|
if (-not (Test-Path -Path Registry::HKEY_CLASSES_ROOT\Msi.Package\shell\Extract\Command))
|
|
{
|
|
New-Item -Path Registry::HKEY_CLASSES_ROOT\Msi.Package\shell\Extract\Command -Force
|
|
}
|
|
$Value = "{0}" -f "msiexec.exe /a `"%1`" /qb TARGETDIR=`"%1 extracted`""
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Msi.Package\shell\Extract\Command -Name "(default)" -PropertyType String -Value $Value -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Msi.Package\shell\Extract -Name MUIVerb -PropertyType String -Value "@shell32.dll,-37514" -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Msi.Package\shell\Extract -Name Icon -PropertyType String -Value "shell32.dll,-16817" -Force
|
|
}
|
|
"Hide"
|
|
{
|
|
Remove-Item -Path Registry::HKEY_CLASSES_ROOT\Msi.Package\shell\Extract -Recurse -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Install" item for the Cabinet (.cab) filenames extensions context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Install" item in the Cabinet (.cab) filenames extensions context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Install" item from the Cabinet (.cab) filenames extensions context menu
|
|
|
|
.EXAMPLE
|
|
CABInstallContext -Show
|
|
|
|
.EXAMPLE
|
|
CABInstallContext -Hide
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function CABInstallContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Show"
|
|
{
|
|
if (-not (Test-Path -Path Registry::HKEY_CLASSES_ROOT\CABFolder\Shell\runas\Command))
|
|
{
|
|
New-Item -Path Registry::HKEY_CLASSES_ROOT\CABFolder\Shell\runas\Command -Force
|
|
}
|
|
$Value = "{0}" -f "cmd /c DISM.exe /Online /Add-Package /PackagePath:`"%1`" /NoRestart & pause"
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\CABFolder\Shell\runas\Command -Name "(default)" -PropertyType String -Value $Value -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\CABFolder\Shell\runas -Name MUIVerb -PropertyType String -Value "@shell32.dll,-10210" -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\CABFolder\Shell\runas -Name HasLUAShield -PropertyType String -Value "" -Force
|
|
}
|
|
"Hide"
|
|
{
|
|
Remove-Item -Path Registry::HKEY_CLASSES_ROOT\CABFolder\Shell\runas -Recurse -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Run as different user" item for the .exe filename extensions context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Run as different user" item in the .exe filename extensions context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Run as different user" item from the .exe filename extensions context menu
|
|
|
|
.EXAMPLE
|
|
RunAsDifferentUserContext -Show
|
|
|
|
.EXAMPLE
|
|
RunAsDifferentUserContext -Hide
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function RunAsDifferentUserContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Show"
|
|
{
|
|
Remove-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\exefile\shell\runasuser -Name Extended -Force -ErrorAction Ignore
|
|
}
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\exefile\shell\runasuser -Name Extended -PropertyType String -Value "" -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Cast to Device" item in the media files and folders context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Cast to Device" item from the media files and folders context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Cast to Device" item in the media files and folders context menu
|
|
|
|
.EXAMPLE
|
|
CastToDeviceContext -Hide
|
|
|
|
.EXAMPLE
|
|
CastToDeviceContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function CastToDeviceContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
if (-not (Test-Path -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Shell Extensions\Blocked"))
|
|
{
|
|
New-Item -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Shell Extensions\Blocked" -Force
|
|
}
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Shell Extensions\Blocked" -Name "{7AD84985-87B4-4a16-BE58-8B72A5B390F7}" -PropertyType String -Value "Play to menu" -Force
|
|
}
|
|
"Show"
|
|
{
|
|
Remove-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Shell Extensions\Blocked" -Name "{7AD84985-87B4-4a16-BE58-8B72A5B390F7}" -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Share" item in the context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Share" item from the context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Share" item in the context menu
|
|
|
|
.EXAMPLE
|
|
ShareContext -Hide
|
|
|
|
.EXAMPLE
|
|
ShareContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function ShareContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
Remove-Item -Path "Registry::HKEY_CLASSES_ROOT\AllFilesystemObjects\shellex\ContextMenuHandlers\ModernSharing" -Recurse -Force -ErrorAction Ignore
|
|
}
|
|
"Show"
|
|
{
|
|
if (-not (Test-Path -Path "Registry::HKEY_CLASSES_ROOT\AllFilesystemObjects\shellex\ContextMenuHandlers\ModernSharing"))
|
|
{
|
|
New-Item -Path "Registry::HKEY_CLASSES_ROOT\AllFilesystemObjects\shellex\ContextMenuHandlers\ModernSharing" -Force
|
|
}
|
|
New-ItemProperty -Path "Registry::HKEY_CLASSES_ROOT\AllFilesystemObjects\shellex\ContextMenuHandlers\ModernSharing" -Name "(default)" -PropertyType String -Value "{e2bf9676-5f8f-435c-97eb-11607a5bedf7}" -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Edit with Photos" item in the media files context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Edit with Photos" item from the media files context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Edit with Photos" item in the media files context menu
|
|
|
|
.EXAMPLE
|
|
EditWithPhotosContext -Hide
|
|
|
|
.EXAMPLE
|
|
EditWithPhotosContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function EditWithPhotosContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.Windows.Photos)
|
|
{
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\AppX43hnxtbyyps62jhe9sqpdzxn1790zetc\Shell\ShellEdit -Name ProgrammaticAccessOnly -PropertyType String -Value "" -Force
|
|
}
|
|
}
|
|
"Show"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.Windows.Photos)
|
|
{
|
|
Remove-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\AppX43hnxtbyyps62jhe9sqpdzxn1790zetc\Shell\ShellEdit -Name ProgrammaticAccessOnly -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Create a new video" item in the media files context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Create a new video" item from the media files context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Create a new video" item in the media files context menu
|
|
|
|
.EXAMPLE
|
|
CreateANewVideoContext -Hide
|
|
|
|
.EXAMPLE
|
|
CreateANewVideoContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function CreateANewVideoContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.Windows.Photos)
|
|
{
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\AppX43hnxtbyyps62jhe9sqpdzxn1790zetc\Shell\ShellCreateVideo -Name ProgrammaticAccessOnly -PropertyType String -Value "" -Force
|
|
}
|
|
}
|
|
"Show"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.Windows.Photos)
|
|
{
|
|
Remove-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\AppX43hnxtbyyps62jhe9sqpdzxn1790zetc\Shell\ShellCreateVideo -Name ProgrammaticAccessOnly -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Print" item in the .bat and .cmd context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Print" item from the .bat and .cmd context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Print" item in the .bat and .cmd context menu
|
|
|
|
.EXAMPLE
|
|
PrintCMDContext -Hide
|
|
|
|
.EXAMPLE
|
|
PrintCMDContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function PrintCMDContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\batfile\shell\print -Name ProgrammaticAccessOnly -PropertyType String -Value "" -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\cmdfile\shell\print -Name ProgrammaticAccessOnly -PropertyType String -Value "" -Force
|
|
}
|
|
"Show"
|
|
{
|
|
Remove-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\batfile\shell\print -Name ProgrammaticAccessOnly -Force -ErrorAction Ignore
|
|
Remove-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\cmdfile\shell\print -Name ProgrammaticAccessOnly -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Include in Library" item in the folders and drives context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Include in Library" item from the folders and drives context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Include in Library" item in the folders and drives context menu
|
|
|
|
.EXAMPLE
|
|
IncludeInLibraryContext -Hide
|
|
|
|
.EXAMPLE
|
|
IncludeInLibraryContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function IncludeInLibraryContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path "Registry::HKEY_CLASSES_ROOT\Folder\ShellEx\ContextMenuHandlers\Library Location" -Name "(default)" -PropertyType String -Value "-{3dad6c5d-2167-4cae-9914-f99e41c12cfa}" -Force
|
|
}
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path "Registry::HKEY_CLASSES_ROOT\Folder\ShellEx\ContextMenuHandlers\Library Location" -Name "(default)" -PropertyType String -Value "{3dad6c5d-2167-4cae-9914-f99e41c12cfa}" -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Send to" item in the folders context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Send to" item from the folders context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Send to" item in the folders context menu
|
|
|
|
.EXAMPLE
|
|
SendToContext -Hide
|
|
|
|
.EXAMPLE
|
|
SendToContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function SendToContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\AllFilesystemObjects\shellex\ContextMenuHandlers\SendTo -Name "(default)" -PropertyType String -Value "-{7BA4C740-9E81-11CF-99D3-00AA004AE837}" -Force
|
|
}
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\AllFilesystemObjects\shellex\ContextMenuHandlers\SendTo -Name "(default)" -PropertyType String -Value "{7BA4C740-9E81-11CF-99D3-00AA004AE837}" -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Turn on BitLocker" item in the drives context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Turn on BitLocker" item from the drives context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Turn on BitLocker" item in the drives context menu
|
|
|
|
.EXAMPLE
|
|
BitLockerContext -Hide
|
|
|
|
.EXAMPLE
|
|
BitLockerContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function BitLockerContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
if (Get-WindowsEdition -Online | Where-Object -FilterScript {($_.Edition -eq "Professional") -or ($_.Edition -like "Enterprise*")})
|
|
{
|
|
if ((Get-BitLockerVolume).ProtectionStatus -eq "Off")
|
|
{
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Drive\shell\encrypt-bde-elev -Name ProgrammaticAccessOnly -PropertyType String -Value "" -Force
|
|
}
|
|
}
|
|
}
|
|
"Show"
|
|
{
|
|
if (Get-WindowsEdition -Online | Where-Object -FilterScript {$_.Edition -eq "Professional" -or $_.Edition -like "Enterprise*"})
|
|
{
|
|
Remove-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Drive\shell\encrypt-bde-elev -Name ProgrammaticAccessOnly -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Compressed (zipped) Folder" item in the "New" context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Compressed (zipped) Folder" item from the "New" context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Compressed (zipped) Folder" item to the "New" context menu
|
|
|
|
.EXAMPLE
|
|
CompressedFolderNewContext -Hide
|
|
|
|
.EXAMPLE
|
|
CompressedFolderNewContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function CompressedFolderNewContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
Remove-Item -Path Registry::HKEY_CLASSES_ROOT\.zip\CompressedFolder\ShellNew -Force -ErrorAction Ignore
|
|
}
|
|
"Show"
|
|
{
|
|
if (-not (Test-Path -Path Registry::HKEY_CLASSES_ROOT\.zip\CompressedFolder\ShellNew))
|
|
{
|
|
New-Item -Path Registry::HKEY_CLASSES_ROOT\.zip\CompressedFolder\ShellNew -Force
|
|
}
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\.zip\CompressedFolder\ShellNew -Name Data -PropertyType Binary -Value ([byte[]](80,75,5,6,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0,0)) -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\.zip\CompressedFolder\ShellNew -Name ItemName -PropertyType ExpandString -Value "@%SystemRoot%\system32\zipfldr.dll,-10194" -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Open", "Print", and "Edit" items if more than 15 files selected
|
|
|
|
.PARAMETER Enable
|
|
Enable the "Open", "Print", and "Edit" items if more than 15 files selected
|
|
|
|
.PARAMETER Disable
|
|
Disable the "Open", "Print", and "Edit" items if more than 15 files selected
|
|
|
|
.EXAMPLE
|
|
MultipleInvokeContext -Enable
|
|
|
|
.EXAMPLE
|
|
MultipleInvokeContext -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function MultipleInvokeContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer -Name MultipleInvokePromptMinimum -PropertyType DWord -Value 300 -Force
|
|
}
|
|
"Disable"
|
|
{
|
|
Remove-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer -Name MultipleInvokePromptMinimum -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Look for an app in the Microsoft Store" item in the "Open with" dialog
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Look for an app in the Microsoft Store" item in the "Open with" dialog
|
|
|
|
.PARAMETER Show
|
|
Show the "Look for an app in the Microsoft Store" item in the "Open with" dialog
|
|
|
|
.EXAMPLE
|
|
UseStoreOpenWith -Hide
|
|
|
|
.EXAMPLE
|
|
UseStoreOpenWith -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function UseStoreOpenWith
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Policies\Microsoft\Windows\Explorer))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name NoUseStoreOpenWith -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
Remove-ItemProperty -Path HKCU:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name NoUseStoreOpenWith -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Open in Windows Terminal" item in the folders context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Open in Windows Terminal" item in the folders context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Open in Windows Terminal" item in the folders context menu
|
|
|
|
.EXAMPLE
|
|
OpenWindowsTerminalContext -Hide
|
|
|
|
.EXAMPLE
|
|
OpenWindowsTerminalContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function OpenWindowsTerminalContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.WindowsTerminal)
|
|
{
|
|
if (-not (Test-Path -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Shell Extensions\Blocked"))
|
|
{
|
|
New-Item -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Shell Extensions\Blocked" -Force
|
|
}
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Shell Extensions\Blocked" -Name "{9F156763-7844-4DC4-B2B1-901F640F5155}" -PropertyType String -Value "WindowsTerminal" -Force
|
|
}
|
|
}
|
|
"Show"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.WindowsTerminal)
|
|
{
|
|
Remove-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Shell Extensions\Blocked" -Name "{9F156763-7844-4DC4-B2B1-901F640F5155}" -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Open in Windows Terminal" (Admin) item in the Desktop and folders context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Open in Windows Terminal" (Admin) item in the Desktop and folders context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Open in Windows Terminal" (Admin) item from the Desktop and folders context menu
|
|
|
|
.EXAMPLE
|
|
OpenWindowsTerminalAdminContext -Show
|
|
|
|
.EXAMPLE
|
|
OpenWindowsTerminalAdminContext -Hide
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function OpenWindowsTerminalAdminContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Show"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.WindowsTerminal)
|
|
{
|
|
# Show the option in the Desktop context menu
|
|
if (-not (Test-Path -Path Registry::HKEY_CLASSES_ROOT\Directory\Background\shell\runas\command))
|
|
{
|
|
New-Item -Path Registry::HKEY_CLASSES_ROOT\Directory\Background\shell\runas\command -ItemType Directory -Force
|
|
}
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Directory\Background\shell\runas -Name "(default)" -PropertyType String -Value $Localization.OpenInWindowsTerminalAdmin -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Directory\Background\shell\runas -Name Icon -PropertyType String -Value "imageres.dll,73" -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Directory\Background\shell\runas -Name NoWorkingDirectory -PropertyType String -Value "" -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Directory\Background\shell\runas\command -Name "(default)" -PropertyType String -Value "wt.exe -d ""%V\.""" -Force
|
|
|
|
# Show the option in the folders context menu
|
|
if (-not (Test-Path -Path Registry::HKEY_CLASSES_ROOT\Directory\shell\runas\command))
|
|
{
|
|
New-Item -Path Registry::HKEY_CLASSES_ROOT\Directory\shell\runas\command -ItemType Directory -Force
|
|
}
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Directory\shell\runas -Name "(default)" -PropertyType String -Value $Localization.OpenInWindowsTerminalAdmin -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Directory\shell\runas -Name Icon -PropertyType String -Value "imageres.dll,73" -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Directory\shell\runas -Name NoWorkingDirectory -PropertyType String -Value "" -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\Directory\shell\runas\command -Name "(default)" -PropertyType String -Value "wt.exe -d ""%1""" -Force
|
|
}
|
|
}
|
|
"Hide"
|
|
{
|
|
$Items = @(
|
|
"Registry::HKEY_CLASSES_ROOT\Directory\Background\shell\runas",
|
|
"Registry::HKEY_CLASSES_ROOT\Directory\shell\runas"
|
|
)
|
|
Remove-Item -Path $Items -Recurse -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Show more options" in the context menu
|
|
|
|
.PARAMETER Enable
|
|
Enable the Windows 10 context menu style
|
|
|
|
.PARAMETER Disable
|
|
Disable the Windows 10 context menu style
|
|
|
|
.EXAMPLE
|
|
Windows10ContextMenu -Enable
|
|
|
|
.EXAMPLE
|
|
Windows10ContextMenu -Disable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function Windows10ContextMenu
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
Remove-Item -Path "HKCU:\Software\Classes\CLSID\{86ca1aa0-34aa-4e8b-a509-50c905bae2a2}" -Recurse -Force -ErrorAction Ignore
|
|
}
|
|
"Enable"
|
|
{
|
|
if (-not (Test-Path -Path "HKCU:\Software\Classes\CLSID\{86ca1aa0-34aa-4e8b-a509-50c905bae2a2}\InprocServer32"))
|
|
{
|
|
New-Item -Path "HKCU:\Software\Classes\CLSID\{86ca1aa0-34aa-4e8b-a509-50c905bae2a2}\InprocServer32" -ItemType Directory -Force
|
|
}
|
|
New-ItemProperty -Path "HKCU:\Software\Classes\CLSID\{86ca1aa0-34aa-4e8b-a509-50c905bae2a2}\InprocServer32" -Name "(default)" -PropertyType String -Value "" -Force
|
|
}
|
|
}
|
|
}
|
|
#endregion Context menu
|
|
|
|
#region Update Policies
|
|
<#
|
|
.SYNOPSIS
|
|
Update Local Group Policy Editor (gpedit.msc) to make all manually created policy keys in the registry visible in the snap-in
|
|
|
|
.EXAMPLE
|
|
UpdateLGPEPolicies
|
|
|
|
.NOTES
|
|
Uses PolicyFileEditor module created by Dave Wyatt
|
|
|
|
.LINK
|
|
https://www.powershellgallery.com/packages/PolicyFileEditor
|
|
|
|
.LINK
|
|
https://github.com/dlwyatt/PolicyFileEditor
|
|
|
|
.NOTES
|
|
Machine-wide user
|
|
#>
|
|
function UpdateLGPEPolicies
|
|
{
|
|
if (-not (Get-WindowsEdition -Online | Where-Object -FilterScript {
|
|
($_.Edition -eq "Professional") -or ($_.Edition -like "Enterprise*") -or ($_.Edition -eq "Education")
|
|
}))
|
|
{
|
|
return
|
|
}
|
|
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
|
|
# Local Machine policies paths to scan recursively
|
|
$LM_Paths = @(
|
|
"HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies",
|
|
"HKLM:\SOFTWARE\Policies\Microsoft\Windows"
|
|
)
|
|
foreach ($Path in (@(Get-ChildItem -Path $LM_Paths -Recurse -Force)))
|
|
{
|
|
foreach ($Item in $Path.Property)
|
|
{
|
|
# Check if property isn't equal to "(default)" and exists
|
|
if (($null -ne $Item) -and ($Item -ne "(default)"))
|
|
{
|
|
# Where all ADMX templates are located to compare with
|
|
foreach ($admx in @(Get-ChildItem -Path "$env:SystemRoot\PolicyDefinitions" -File -Force))
|
|
{
|
|
# Parse every ADMX template searching if it contains full path and registry key simultaneously
|
|
[xml]$config = Get-Content -Path $admx.FullName -Encoding UTF8
|
|
$SplitPath = Split-Path -Path $Path.Name.Replace("HKEY_LOCAL_MACHINE\", "HKLM:") -NoQualifier
|
|
|
|
if ($config.policyDefinitions.policies.policy | Where-Object -FilterScript {
|
|
($_.key -eq $SplitPath) -and ($_.valueName -eq $Item) -or (($_.key -eq $SplitPath) -and ($_.Name -eq $Item))
|
|
})
|
|
{
|
|
try
|
|
{
|
|
Write-Verbose -Message $Item.Replace("{}", "") -Verbose
|
|
|
|
$Parameters = @{
|
|
Path = "$env:SystemRoot\System32\GroupPolicy\Machine\Registry.pol"
|
|
# e.g. SOFTWARE\Microsoft\Windows\CurrentVersion\Policies
|
|
Key = Split-Path -Path $Path.Name.Replace("HKEY_LOCAL_MACHINE\", "HKLM:") -NoQualifier
|
|
ValueName = $Item.Replace("{}", "")
|
|
Data = Get-ItemPropertyValue -Path $Path.PSPath -Name $Item
|
|
# DWord, String, etc.
|
|
Type = (Get-Item -Path $Path.PSPath).GetValueKind($Item)
|
|
# Do not update the policy DB every time
|
|
NoGptIniUpdate = $true
|
|
}
|
|
Set-PolicyFileEntry @Parameters
|
|
}
|
|
catch
|
|
{}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# Current User policies paths to scan recursively
|
|
$CU_Paths = @(
|
|
"HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies",
|
|
"HKCU:\SOFTWARE\Policies\Microsoft\Windows"
|
|
)
|
|
foreach ($Path in (@(Get-ChildItem -Path $CU_Paths -Recurse -Force)))
|
|
{
|
|
foreach ($Item in $Path.Property)
|
|
{
|
|
# Check if property isn't equal to "(default)" and exists
|
|
if (($null -ne $Item) -and ($Item -ne "(default)"))
|
|
{
|
|
# Where all ADMX templates are located to compare with
|
|
foreach ($admx in @(Get-ChildItem -Path "$env:SystemRoot\PolicyDefinitions" -File -Force))
|
|
{
|
|
# Parse every ADMX template searching if it contains full path and registry key simultaneously
|
|
[xml]$config = Get-Content -Path $admx.FullName -Encoding UTF8
|
|
$SplitPath = Split-Path -Path $Path.Name.Replace("HKEY_CURRENT_USER\", "HKCU:") -NoQualifier
|
|
|
|
if ($config.policyDefinitions.policies.policy | Where-Object -FilterScript {
|
|
($_.key -eq $SplitPath) -and ($_.valueName -eq $Item) -or (($_.key -eq $SplitPath) -and ($_.Name -eq $Item))
|
|
})
|
|
{
|
|
try
|
|
{
|
|
Write-Verbose -Message $Item.Replace("{}", "") -Verbose
|
|
|
|
$Parameters = @{
|
|
Path = "$env:SystemRoot\System32\GroupPolicy\User\Registry.pol"
|
|
# e.g. SOFTWARE\Microsoft\Windows\CurrentVersion\Policies
|
|
Key = Split-Path -Path $Path.Name.Replace("HKEY_CURRENT_USER\", "HKCU:") -NoQualifier
|
|
ValueName = $Item.Replace("{}", "")
|
|
Data = Get-ItemPropertyValue -Path $Path.PSPath -Name $Item
|
|
# DWord, String, etc.
|
|
Type = (Get-Item -Path $Path.PSPath).GetValueKind($Item)
|
|
# Do not update the policy DB every time
|
|
NoGptIniUpdate = $true
|
|
}
|
|
Set-PolicyFileEntry @Parameters
|
|
}
|
|
catch
|
|
{}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# Re-build GPT.ini if it doesn't exist
|
|
if (-not (Test-Path -Path $env:SystemRoot\System32\GroupPolicy\GPT.ini))
|
|
{
|
|
Start-Process -FilePath gpedit.msc
|
|
Start-Sleep -Seconds 2
|
|
|
|
# Get mmc.exe's Id with its' argument (gpedit.msc) to close
|
|
$gpedit_Process_ID = (Get-CimInstance -ClassName CIM_Process | Where-Object -FilterScript {
|
|
$_.Name -eq "mmc.exe"
|
|
} | Where-Object -FilterScript {$_.CommandLine -match "GPEDIT.MSC"}).Handle
|
|
Get-Process -Id $gpedit_Process_ID | Stop-Process -Force
|
|
}
|
|
|
|
Update-GptIniVersion -Path $env:SystemRoot\System32\GroupPolicy\GPT.ini -PolicyType Machine, User
|
|
|
|
# Apply the new policy immediately
|
|
gpupdate.exe /force
|
|
}
|
|
#endregion Update Policies
|
|
|
|
#region Refresh Environment
|
|
function RefreshEnvironment
|
|
{
|
|
$UpdateEnvironment = @{
|
|
Namespace = "WinAPI"
|
|
Name = "UpdateEnvironment"
|
|
Language = "CSharp"
|
|
MemberDefinition = @"
|
|
private static readonly IntPtr HWND_BROADCAST = new IntPtr(0xffff);
|
|
private const int WM_SETTINGCHANGE = 0x1a;
|
|
private const int SMTO_ABORTIFHUNG = 0x0002;
|
|
|
|
[DllImport("shell32.dll", CharSet = CharSet.Auto, SetLastError = false)]
|
|
private static extern int SHChangeNotify(int eventId, int flags, IntPtr item1, IntPtr item2);
|
|
|
|
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = false)]
|
|
private static extern IntPtr SendMessageTimeout(IntPtr hWnd, int Msg, IntPtr wParam, string lParam, int fuFlags, int uTimeout, IntPtr lpdwResult);
|
|
|
|
[DllImport("user32.dll", CharSet = CharSet.Auto, SetLastError = false)]
|
|
static extern bool SendNotifyMessage(IntPtr hWnd, uint Msg, IntPtr wParam, string lParam);
|
|
|
|
public static void Refresh()
|
|
{
|
|
// Update desktop icons
|
|
SHChangeNotify(0x8000000, 0x1000, IntPtr.Zero, IntPtr.Zero);
|
|
|
|
// Update environment variables
|
|
SendMessageTimeout(HWND_BROADCAST, WM_SETTINGCHANGE, IntPtr.Zero, null, SMTO_ABORTIFHUNG, 100, IntPtr.Zero);
|
|
|
|
// Update taskbar
|
|
SendNotifyMessage(HWND_BROADCAST, WM_SETTINGCHANGE, IntPtr.Zero, "TraySettings");
|
|
}
|
|
|
|
private static readonly IntPtr hWnd = new IntPtr(65535);
|
|
private const int Msg = 273;
|
|
// Virtual key ID of the F5 in File Explorer
|
|
private static readonly UIntPtr UIntPtr = new UIntPtr(41504);
|
|
|
|
[DllImport("user32.dll", SetLastError=true)]
|
|
public static extern int PostMessageW(IntPtr hWnd, uint Msg, UIntPtr wParam, IntPtr lParam);
|
|
|
|
public static void PostMessage()
|
|
{
|
|
// Simulate pressing F5 to refresh the desktop
|
|
PostMessageW(hWnd, Msg, UIntPtr, IntPtr.Zero);
|
|
}
|
|
"@
|
|
}
|
|
if (-not ("WinAPI.UpdateEnvironment" -as [type]))
|
|
{
|
|
Add-Type @UpdateEnvironment
|
|
}
|
|
|
|
# Simulate pressing F5 to refresh the desktop
|
|
[WinAPI.UpdateEnvironment]::PostMessage()
|
|
|
|
# Refresh desktop icons, environment variables, taskbar
|
|
[WinAPI.UpdateEnvironment]::Refresh()
|
|
|
|
# Restart the Start menu
|
|
Stop-Process -Name StartMenuExperienceHost -Force -ErrorAction Ignore
|
|
|
|
# Turn on Controlled folder access if it was turned off
|
|
if ($Script:DefenderEnabled)
|
|
{
|
|
if ($Script:ControlledFolderAccess)
|
|
{
|
|
Set-MpPreference -EnableControlledFolderAccess Enabled
|
|
}
|
|
}
|
|
|
|
if ($Script:RegionChanged)
|
|
{
|
|
# Set the original region ID
|
|
Set-WinHomeLocation -GeoId $Script:Region
|
|
}
|
|
|
|
# In order for the changes to take effect the File Explorer process has to be restarted
|
|
$Title = ""
|
|
$Message = $Localization.FileExplorerRestartPrompt
|
|
$Yes = $Localization.Yes
|
|
$No = $Localization.No
|
|
$Options = "&$No", "&$Yes"
|
|
$DefaultChoice = 1
|
|
$Result = $Host.UI.PromptForChoice($Title, $Message, $Options, $DefaultChoice)
|
|
|
|
switch ($Result)
|
|
{
|
|
"0"
|
|
{
|
|
continue
|
|
}
|
|
"1"
|
|
{
|
|
Stop-Process -Name explorer -Force
|
|
|
|
Start-Sleep -Seconds 3
|
|
|
|
# Restoring closed folders
|
|
foreach ($Script:OpenedFolder in $Script:OpenedFolders)
|
|
{
|
|
if (Test-Path -Path $Script:OpenedFolder)
|
|
{
|
|
Start-Process -FilePath explorer -ArgumentList $Script:OpenedFolder
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Warning -Message $Localization.RestartWarning
|
|
|
|
[Windows.UI.Notifications.ToastNotificationManager, Windows.UI.Notifications, ContentType = WindowsRuntime] | Out-Null
|
|
[Windows.Data.Xml.Dom.XmlDocument, Windows.Data.Xml.Dom.XmlDocument, ContentType = WindowsRuntime] | Out-Null
|
|
|
|
# Telegram group
|
|
[xml]$ToastTemplate = @"
|
|
<toast duration="Long" scenario="reminder">
|
|
<visual>
|
|
<binding template="ToastGeneric">
|
|
<text>$($Localization.TelegramGroupTitle)</text>
|
|
<group>
|
|
<subgroup>
|
|
<text hint-style="body" hint-wrap="true">https://t.me/sophia_chat</text>
|
|
</subgroup>
|
|
</group>
|
|
</binding>
|
|
</visual>
|
|
<audio src="ms-winsoundevent:notification.default" />
|
|
<actions>
|
|
<action arguments="https://t.me/sophia_chat" content="$($Localization.Open)" activationType="protocol"/>
|
|
<action arguments="dismiss" content="" activationType="system"/>
|
|
</actions>
|
|
</toast>
|
|
"@
|
|
|
|
$ToastXml = [Windows.Data.Xml.Dom.XmlDocument]::New()
|
|
$ToastXml.LoadXml($ToastTemplate.OuterXml)
|
|
|
|
$ToastMessage = [Windows.UI.Notifications.ToastNotification]::New($ToastXML)
|
|
[Windows.UI.Notifications.ToastNotificationManager]::CreateToastNotifier("windows.immersivecontrolpanel_cw5n1h2txyewy!microsoft.windows.immersivecontrolpanel").Show($ToastMessage)
|
|
|
|
# Telegram channel
|
|
[xml]$ToastTemplate = @"
|
|
<toast duration="Long" scenario="reminder">
|
|
<visual>
|
|
<binding template="ToastGeneric">
|
|
<text>$($Localization.TelegramChannelTitle)</text>
|
|
<group>
|
|
<subgroup>
|
|
<text hint-style="body" hint-wrap="true">https://t.me/sophianews</text>
|
|
</subgroup>
|
|
</group>
|
|
</binding>
|
|
</visual>
|
|
<audio src="ms-winsoundevent:notification.default" />
|
|
<actions>
|
|
<action arguments="https://t.me/sophianews" content="$($Localization.Open)" activationType="protocol"/>
|
|
<action arguments="dismiss" content="" activationType="system"/>
|
|
</actions>
|
|
</toast>
|
|
"@
|
|
|
|
$ToastXml = [Windows.Data.Xml.Dom.XmlDocument]::New()
|
|
$ToastXml.LoadXml($ToastTemplate.OuterXml)
|
|
|
|
$ToastMessage = [Windows.UI.Notifications.ToastNotification]::New($ToastXML)
|
|
[Windows.UI.Notifications.ToastNotificationManager]::CreateToastNotifier("windows.immersivecontrolpanel_cw5n1h2txyewy!microsoft.windows.immersivecontrolpanel").Show($ToastMessage)
|
|
}
|
|
#endregion Refresh Environment
|
|
|
|
# Errors output
|
|
function Errors
|
|
{
|
|
if ($Global:Error)
|
|
{
|
|
($Global:Error | ForEach-Object -Process {
|
|
# Some errors may have the Windows nature and don't have a path to any of the module's files
|
|
$ErrorInFile = if ($_.InvocationInfo.PSCommandPath)
|
|
{
|
|
Split-Path -Path $_.InvocationInfo.PSCommandPath -Leaf
|
|
}
|
|
|
|
[PSCustomObject]@{
|
|
$Localization.ErrorsLine = $_.InvocationInfo.ScriptLineNumber
|
|
$Localization.ErrorsFile = $ErrorInFile
|
|
$Localization.ErrorsMessage = $_.Exception.Message
|
|
}
|
|
} | Sort-Object -Property Line | Format-Table -AutoSize -Wrap | Out-String).Trim()
|
|
}
|
|
}
|
|
|
|
# SIG # Begin signature block
|
|
# MIIbvwYJKoZIhvcNAQcCoIIbsDCCG6wCAQExCzAJBgUrDgMCGgUAMGkGCisGAQQB
|
|
# gjcCAQSgWzBZMDQGCisGAQQBgjcCAR4wJgIDAQAABBAfzDtgWUsITrck0sYpfvNR
|
|
# AgEAAgEAAgEAAgEAAgEAMCEwCQYFKw4DAhoFAAQUf1jTmGGjSC+KdppJDHzCpj5i
|
|
# 2r+gghY3MIIDAjCCAeqgAwIBAgIQbPiUYMntEr9PH9C5Mau3xTANBgkqhkiG9w0B
|
|
# AQsFADAZMRcwFQYDVQQDDA5Tb3BoaWEgUHJvamVjdDAeFw0yMjA3MTYxNDM2MDda
|
|
# Fw0yNDA3MTYxNDQ2MDdaMBkxFzAVBgNVBAMMDlNvcGhpYSBQcm9qZWN0MIIBIjAN
|
|
# BgkqhkiG9w0BAQEFAAOCAQ8AMIIBCgKCAQEAtFM5b/hNUQPcNCKlWVisE5zIERdC
|
|
# TZWdsnkliCg98qv0qkPH08AAuZ5LOgdctfaocVX4RbygcWCLyaphktSC29aMHmP/
|
|
# T/JPBSYnLmWIJ6NbNVBn4BqqCZusjmYRj1LH+Ega9Kv4gFfF4lyCE8OnebDkWPu1
|
|
# Y+5VAoZ6BaVvHEw5rtrHcH8/W15mE1Z+uuYIKNVPg3MMGNGQ5MiLUduuCu5l+J4K
|
|
# yU/nLvx7tI3tIx8wJY2S2SYDNPf/bMOeeb0jlcD1/bVh5TNn3Z4wIXQ3r4yNxllO
|
|
# dBPidPT3Ck4+aDTzP/Y8vrFeIMLKSCYGY+cOCankPZmUCVFJxABvW5EKDQIDAQAB
|
|
# o0YwRDAOBgNVHQ8BAf8EBAMCB4AwEwYDVR0lBAwwCgYIKwYBBQUHAwMwHQYDVR0O
|
|
# BBYEFMJrBhADhZmuEvA18MvGfrBsUpDhMA0GCSqGSIb3DQEBCwUAA4IBAQCzuSeY
|
|
# YdiLsJYFPUW21r2QjSbIi7oOAa58Y2C7KM+QvAm7R6W7l3y6Zz58mBZmSCt+u6au
|
|
# HHYYxo71eCKcH6Lx/2BAeg3GBfZClCZ480nxH1+T4fobdWuT/f4Iw8ZalqcnqJQQ
|
|
# xXhK2U8sMS6oIlb8VZhzosiP68gSZwNgcBbO8P1nAVa7WXyhdmo5CTpa1CihfOT2
|
|
# a3x7runLfgPHmgDcTQpJYAl4xA1JgM+5adlGo33uDXNDgN+6MPZKCuD89wFIYbXw
|
|
# WM6i+SHOBvqACWvM5goBCiVuIq2aI/6r6jeW8UiN51L0/IIniY+P0e7ZibjiadZR
|
|
# DuMQLs5d3GLELtT3MIIFsTCCBJmgAwIBAgIQASQK+x44C4oW8UtxnfTTwDANBgkq
|
|
# hkiG9w0BAQwFADBlMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5j
|
|
# MRkwFwYDVQQLExB3d3cuZGlnaWNlcnQuY29tMSQwIgYDVQQDExtEaWdpQ2VydCBB
|
|
# c3N1cmVkIElEIFJvb3QgQ0EwHhcNMjIwNjA5MDAwMDAwWhcNMzExMTA5MjM1OTU5
|
|
# WjBiMQswCQYDVQQGEwJVUzEVMBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQL
|
|
# ExB3d3cuZGlnaWNlcnQuY29tMSEwHwYDVQQDExhEaWdpQ2VydCBUcnVzdGVkIFJv
|
|
# b3QgRzQwggIiMA0GCSqGSIb3DQEBAQUAA4ICDwAwggIKAoICAQC/5pBzaN675F1K
|
|
# PDAiMGkz7MKnJS7JIT3yithZwuEppz1Yq3aaza57G4QNxDAf8xukOBbrVsaXbR2r
|
|
# snnyyhHS5F/WBTxSD1Ifxp4VpX6+n6lXFllVcq9ok3DCsrp1mWpzMpTREEQQLt+C
|
|
# 8weE5nQ7bXHiLQwb7iDVySAdYyktzuxeTsiT+CFhmzTrBcZe7FsavOvJz82sNEBf
|
|
# sXpm7nfISKhmV1efVFiODCu3T6cw2Vbuyntd463JT17lNecxy9qTXtyOj4DatpGY
|
|
# QJB5w3jHtrHEtWoYOAMQjdjUN6QuBX2I9YI+EJFwq1WCQTLX2wRzKm6RAXwhTNS8
|
|
# rhsDdV14Ztk6MUSaM0C/CNdaSaTC5qmgZ92kJ7yhTzm1EVgX9yRcRo9k98FpiHaY
|
|
# dj1ZXUJ2h4mXaXpI8OCiEhtmmnTK3kse5w5jrubU75KSOp493ADkRSWJtppEGSt+
|
|
# wJS00mFt6zPZxd9LBADMfRyVw4/3IbKyEbe7f/LVjHAsQWCqsWMYRJUadmJ+9oCw
|
|
# ++hkpjPRiQfhvbfmQ6QYuKZ3AeEPlAwhHbJUKSWJbOUOUlFHdL4mrLZBdd56rF+N
|
|
# P8m800ERElvlEFDrMcXKchYiCd98THU/Y+whX8QgUWtvsauGi0/C1kVfnSD8oR7F
|
|
# wI+isX4KJpn15GkvmB0t9dmpsh3lGwIDAQABo4IBXjCCAVowDwYDVR0TAQH/BAUw
|
|
# AwEB/zAdBgNVHQ4EFgQU7NfjgtJxXWRM3y5nP+e6mK4cD08wHwYDVR0jBBgwFoAU
|
|
# Reuir/SSy4IxLVGLp6chnfNtyA8wDgYDVR0PAQH/BAQDAgGGMBMGA1UdJQQMMAoG
|
|
# CCsGAQUFBwMIMHkGCCsGAQUFBwEBBG0wazAkBggrBgEFBQcwAYYYaHR0cDovL29j
|
|
# c3AuZGlnaWNlcnQuY29tMEMGCCsGAQUFBzAChjdodHRwOi8vY2FjZXJ0cy5kaWdp
|
|
# Y2VydC5jb20vRGlnaUNlcnRBc3N1cmVkSURSb290Q0EuY3J0MEUGA1UdHwQ+MDww
|
|
# OqA4oDaGNGh0dHA6Ly9jcmwzLmRpZ2ljZXJ0LmNvbS9EaWdpQ2VydEFzc3VyZWRJ
|
|
# RFJvb3RDQS5jcmwwIAYDVR0gBBkwFzAIBgZngQwBBAIwCwYJYIZIAYb9bAcBMA0G
|
|
# CSqGSIb3DQEBDAUAA4IBAQCaFgKlAe+B+w20WLJ4ragjGdlzN9pgnlHXy/gvQLmj
|
|
# H3xATjM+kDzniQF1hehiex1W4HG63l7GN7x5XGIATfhJelFNBjLzxdIAKicg6oku
|
|
# FTngLD74dXwsgkFhNQ8j0O01ldKIlSlDy+CmWBB8U46fRckgNxTA7Rm6fnc50lSW
|
|
# x6YR3zQz9nVSQkscnY2W1ZVsRxIUJF8mQfoaRr3esOWRRwOsGAjLy9tmiX8rnGW/
|
|
# vjdOvi3znUrDzMxHXsiVla3Ry7sqBiD5P3LqNutFcpJ6KXsUAzz7TdZIcXoQEYoI
|
|
# dM1sGwRc0oqVA3ZRUFPWLvdKRsOuECxxTLCHtic3RGBEMIIGrjCCBJagAwIBAgIQ
|
|
# BzY3tyRUfNhHrP0oZipeWzANBgkqhkiG9w0BAQsFADBiMQswCQYDVQQGEwJVUzEV
|
|
# MBMGA1UEChMMRGlnaUNlcnQgSW5jMRkwFwYDVQQLExB3d3cuZGlnaWNlcnQuY29t
|
|
# MSEwHwYDVQQDExhEaWdpQ2VydCBUcnVzdGVkIFJvb3QgRzQwHhcNMjIwMzIzMDAw
|
|
# MDAwWhcNMzcwMzIyMjM1OTU5WjBjMQswCQYDVQQGEwJVUzEXMBUGA1UEChMORGln
|
|
# aUNlcnQsIEluYy4xOzA5BgNVBAMTMkRpZ2lDZXJ0IFRydXN0ZWQgRzQgUlNBNDA5
|
|
# NiBTSEEyNTYgVGltZVN0YW1waW5nIENBMIICIjANBgkqhkiG9w0BAQEFAAOCAg8A
|
|
# MIICCgKCAgEAxoY1BkmzwT1ySVFVxyUDxPKRN6mXUaHW0oPRnkyibaCwzIP5WvYR
|
|
# oUQVQl+kiPNo+n3znIkLf50fng8zH1ATCyZzlm34V6gCff1DtITaEfFzsbPuK4CE
|
|
# iiIY3+vaPcQXf6sZKz5C3GeO6lE98NZW1OcoLevTsbV15x8GZY2UKdPZ7Gnf2ZCH
|
|
# RgB720RBidx8ald68Dd5n12sy+iEZLRS8nZH92GDGd1ftFQLIWhuNyG7QKxfst5K
|
|
# fc71ORJn7w6lY2zkpsUdzTYNXNXmG6jBZHRAp8ByxbpOH7G1WE15/tePc5OsLDni
|
|
# pUjW8LAxE6lXKZYnLvWHpo9OdhVVJnCYJn+gGkcgQ+NDY4B7dW4nJZCYOjgRs/b2
|
|
# nuY7W+yB3iIU2YIqx5K/oN7jPqJz+ucfWmyU8lKVEStYdEAoq3NDzt9KoRxrOMUp
|
|
# 88qqlnNCaJ+2RrOdOqPVA+C/8KI8ykLcGEh/FDTP0kyr75s9/g64ZCr6dSgkQe1C
|
|
# vwWcZklSUPRR8zZJTYsg0ixXNXkrqPNFYLwjjVj33GHek/45wPmyMKVM1+mYSlg+
|
|
# 0wOI/rOP015LdhJRk8mMDDtbiiKowSYI+RQQEgN9XyO7ZONj4KbhPvbCdLI/Hgl2
|
|
# 7KtdRnXiYKNYCQEoAA6EVO7O6V3IXjASvUaetdN2udIOa5kM0jO0zbECAwEAAaOC
|
|
# AV0wggFZMBIGA1UdEwEB/wQIMAYBAf8CAQAwHQYDVR0OBBYEFLoW2W1NhS9zKXaa
|
|
# L3WMaiCPnshvMB8GA1UdIwQYMBaAFOzX44LScV1kTN8uZz/nupiuHA9PMA4GA1Ud
|
|
# DwEB/wQEAwIBhjATBgNVHSUEDDAKBggrBgEFBQcDCDB3BggrBgEFBQcBAQRrMGkw
|
|
# JAYIKwYBBQUHMAGGGGh0dHA6Ly9vY3NwLmRpZ2ljZXJ0LmNvbTBBBggrBgEFBQcw
|
|
# AoY1aHR0cDovL2NhY2VydHMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0VHJ1c3RlZFJv
|
|
# b3RHNC5jcnQwQwYDVR0fBDwwOjA4oDagNIYyaHR0cDovL2NybDMuZGlnaWNlcnQu
|
|
# Y29tL0RpZ2lDZXJ0VHJ1c3RlZFJvb3RHNC5jcmwwIAYDVR0gBBkwFzAIBgZngQwB
|
|
# BAIwCwYJYIZIAYb9bAcBMA0GCSqGSIb3DQEBCwUAA4ICAQB9WY7Ak7ZvmKlEIgF+
|
|
# ZtbYIULhsBguEE0TzzBTzr8Y+8dQXeJLKftwig2qKWn8acHPHQfpPmDI2AvlXFvX
|
|
# bYf6hCAlNDFnzbYSlm/EUExiHQwIgqgWvalWzxVzjQEiJc6VaT9Hd/tydBTX/6tP
|
|
# iix6q4XNQ1/tYLaqT5Fmniye4Iqs5f2MvGQmh2ySvZ180HAKfO+ovHVPulr3qRCy
|
|
# Xen/KFSJ8NWKcXZl2szwcqMj+sAngkSumScbqyQeJsG33irr9p6xeZmBo1aGqwpF
|
|
# yd/EjaDnmPv7pp1yr8THwcFqcdnGE4AJxLafzYeHJLtPo0m5d2aR8XKc6UsCUqc3
|
|
# fpNTrDsdCEkPlM05et3/JWOZJyw9P2un8WbDQc1PtkCbISFA0LcTJM3cHXg65J6t
|
|
# 5TRxktcma+Q4c6umAU+9Pzt4rUyt+8SVe+0KXzM5h0F4ejjpnOHdI/0dKNPH+ejx
|
|
# mF/7K9h+8kaddSweJywm228Vex4Ziza4k9Tm8heZWcpw8De/mADfIBZPJ/tgZxah
|
|
# ZrrdVcA6KYawmKAr7ZVBtzrVFZgxtGIJDwq9gdkT/r+k0fNX2bwE+oLeMt8EifAA
|
|
# zV3C+dAjfwAL5HYCJtnwZXZCpimHCUcr5n8apIUP/JiW9lVUKx+A+sDyDivl1vup
|
|
# L0QVSucTDh3bNzgaoSv27dZ8/DCCBsYwggSuoAMCAQICEAp6SoieyZlCkAZjOE2G
|
|
# l50wDQYJKoZIhvcNAQELBQAwYzELMAkGA1UEBhMCVVMxFzAVBgNVBAoTDkRpZ2lD
|
|
# ZXJ0LCBJbmMuMTswOQYDVQQDEzJEaWdpQ2VydCBUcnVzdGVkIEc0IFJTQTQwOTYg
|
|
# U0hBMjU2IFRpbWVTdGFtcGluZyBDQTAeFw0yMjAzMjkwMDAwMDBaFw0zMzAzMTQy
|
|
# MzU5NTlaMEwxCzAJBgNVBAYTAlVTMRcwFQYDVQQKEw5EaWdpQ2VydCwgSW5jLjEk
|
|
# MCIGA1UEAxMbRGlnaUNlcnQgVGltZXN0YW1wIDIwMjIgLSAyMIICIjANBgkqhkiG
|
|
# 9w0BAQEFAAOCAg8AMIICCgKCAgEAuSqWI6ZcvF/WSfAVghj0M+7MXGzj4CUu0jHk
|
|
# PECu+6vE43hdflw26vUljUOjges4Y/k8iGnePNIwUQ0xB7pGbumjS0joiUF/DbLW
|
|
# +YTxmD4LvwqEEnFsoWImAdPOw2z9rDt+3Cocqb0wxhbY2rzrsvGD0Z/NCcW5QWpF
|
|
# QiNBWvhg02UsPn5evZan8Pyx9PQoz0J5HzvHkwdoaOVENFJfD1De1FksRHTAMkcZ
|
|
# W+KYLo/Qyj//xmfPPJOVToTpdhiYmREUxSsMoDPbTSSF6IKU4S8D7n+FAsmG4dUY
|
|
# FLcERfPgOL2ivXpxmOwV5/0u7NKbAIqsHY07gGj+0FmYJs7g7a5/KC7CnuALS8gI
|
|
# 0TK7g/ojPNn/0oy790Mj3+fDWgVifnAs5SuyPWPqyK6BIGtDich+X7Aa3Rm9n3RB
|
|
# Cq+5jgnTdKEvsFR2wZBPlOyGYf/bES+SAzDOMLeLD11Es0MdI1DNkdcvnfv8zbHB
|
|
# p8QOxO9APhk6AtQxqWmgSfl14ZvoaORqDI/r5LEhe4ZnWH5/H+gr5BSyFtaBocra
|
|
# MJBr7m91wLA2JrIIO/+9vn9sExjfxm2keUmti39hhwVo99Rw40KV6J67m0uy4rZB
|
|
# Peevpxooya1hsKBBGBlO7UebYZXtPgthWuo+epiSUc0/yUTngIspQnL3ebLdhOon
|
|
# 7v59emsCAwEAAaOCAYswggGHMA4GA1UdDwEB/wQEAwIHgDAMBgNVHRMBAf8EAjAA
|
|
# MBYGA1UdJQEB/wQMMAoGCCsGAQUFBwMIMCAGA1UdIAQZMBcwCAYGZ4EMAQQCMAsG
|
|
# CWCGSAGG/WwHATAfBgNVHSMEGDAWgBS6FtltTYUvcyl2mi91jGogj57IbzAdBgNV
|
|
# HQ4EFgQUjWS3iSH+VlhEhGGn6m8cNo/drw0wWgYDVR0fBFMwUTBPoE2gS4ZJaHR0
|
|
# cDovL2NybDMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0VHJ1c3RlZEc0UlNBNDA5NlNI
|
|
# QTI1NlRpbWVTdGFtcGluZ0NBLmNybDCBkAYIKwYBBQUHAQEEgYMwgYAwJAYIKwYB
|
|
# BQUHMAGGGGh0dHA6Ly9vY3NwLmRpZ2ljZXJ0LmNvbTBYBggrBgEFBQcwAoZMaHR0
|
|
# cDovL2NhY2VydHMuZGlnaWNlcnQuY29tL0RpZ2lDZXJ0VHJ1c3RlZEc0UlNBNDA5
|
|
# NlNIQTI1NlRpbWVTdGFtcGluZ0NBLmNydDANBgkqhkiG9w0BAQsFAAOCAgEADS0j
|
|
# dKbR9fjqS5k/AeT2DOSvFp3Zs4yXgimcQ28BLas4tXARv4QZiz9d5YZPvpM63io5
|
|
# WjlO2IRZpbwbmKrobO/RSGkZOFvPiTkdcHDZTt8jImzV3/ZZy6HC6kx2yqHcoSuW
|
|
# uJtVqRprfdH1AglPgtalc4jEmIDf7kmVt7PMxafuDuHvHjiKn+8RyTFKWLbfOHzL
|
|
# +lz35FO/bgp8ftfemNUpZYkPopzAZfQBImXH6l50pls1klB89Bemh2RPPkaJFmMg
|
|
# a8vye9A140pwSKm25x1gvQQiFSVwBnKpRDtpRxHT7unHoD5PELkwNuTzqmkJqIt+
|
|
# ZKJllBH7bjLx9bs4rc3AkxHVMnhKSzcqTPNc3LaFwLtwMFV41pj+VG1/calIGnjd
|
|
# RncuG3rAM4r4SiiMEqhzzy350yPynhngDZQooOvbGlGglYKOKGukzp123qlzqkhq
|
|
# WUOuX+r4DwZCnd8GaJb+KqB0W2Nm3mssuHiqTXBt8CzxBxV+NbTmtQyimaXXFWs1
|
|
# DoXW4CzM4AwkuHxSCx6ZfO/IyMWMWGmvqz3hz8x9Fa4Uv4px38qXsdhH6hyF4EVO
|
|
# EhwUKVjMb9N/y77BDkpvIJyu2XMyWQjnLZKhGhH+MpimXSuX4IvTnMxttQ2uR2M4
|
|
# RxdbbxPaahBuH0m3RFu0CAqHWlkEdhGhp3cCExwxggTyMIIE7gIBATAtMBkxFzAV
|
|
# BgNVBAMMDlNvcGhpYSBQcm9qZWN0AhBs+JRgye0Sv08f0Lkxq7fFMAkGBSsOAwIa
|
|
# BQCgeDAYBgorBgEEAYI3AgEMMQowCKACgAChAoAAMBkGCSqGSIb3DQEJAzEMBgor
|
|
# BgEEAYI3AgEEMBwGCisGAQQBgjcCAQsxDjAMBgorBgEEAYI3AgEVMCMGCSqGSIb3
|
|
# DQEJBDEWBBQL1uSSPwEQ3eKWisyh56mLW4iNuDANBgkqhkiG9w0BAQEFAASCAQBA
|
|
# D2tYhaG9doHMa5VD/X2MHyJnf+Fd0JnYS0t58cMhsScONsbxlCK+fy7ktqhCEk4e
|
|
# leL34XsbraUv0Bsg7JA6QX2y8oloVurgb+acl+Kj9hPuLxwSPAC2PPoTLjbHL2mN
|
|
# R//qB5khNW3l7BEzySRHBwy7ftGjI4xBTwoJkyabV/vnrF1vai1gCh9++A/0gHzj
|
|
# SO/mqCgQGJ7yzn/uL0YWtn6/CZCB84rsS9jdKsHMzcG1utA12ISQYvRmjmClMU44
|
|
# mmOMsoiQygXI6xnitEDhwhuxqyKv+kSB9Olfizo3Ua/A2CCTmmbCozmiUM5UCi7R
|
|
# yar7vCLrDyE1o3lsJQzAoYIDIDCCAxwGCSqGSIb3DQEJBjGCAw0wggMJAgEBMHcw
|
|
# YzELMAkGA1UEBhMCVVMxFzAVBgNVBAoTDkRpZ2lDZXJ0LCBJbmMuMTswOQYDVQQD
|
|
# EzJEaWdpQ2VydCBUcnVzdGVkIEc0IFJTQTQwOTYgU0hBMjU2IFRpbWVTdGFtcGlu
|
|
# ZyBDQQIQCnpKiJ7JmUKQBmM4TYaXnTANBglghkgBZQMEAgEFAKBpMBgGCSqGSIb3
|
|
# DQEJAzELBgkqhkiG9w0BBwEwHAYJKoZIhvcNAQkFMQ8XDTIyMDcxNjE0NDYyMVow
|
|
# LwYJKoZIhvcNAQkEMSIEIGUTNxvTh5CoUSsfNpw4NXpD3aSWlmQWLWGyzD/KFhg7
|
|
# MA0GCSqGSIb3DQEBAQUABIICADNrnSyFO59nhGv0VRq/JhWUNKaKUhnZjXbdrMZh
|
|
# DY+NLhqDqbkkJK+nwZw8g2jI+B/L0NAf3e1TkTbYuDSg0miPesUcnzUPUvY508Sq
|
|
# R3U0C6vxF12GxJC8P9MjmIJLk08tIZHVlXGTiOFFg7ygXSNa9cBmP6J1C6tbWeoZ
|
|
# iBBXIy+JRSyUENM2GhpvqMPVwociL6UW5vaMOJEje/LnYVGorsDcEzs0xzW/i2Zp
|
|
# ZmLBSMvIJlVY9IMEVLkximjP0rR9WhV98OrAe8Aho89yFSbaHU4AMtxMumssQcd8
|
|
# 1EhyqNeT4jv/J5Q9j4/3RA+T4fV66R3KQQOkke53fbXtv9R7eWW6YVNLsPI7B5HF
|
|
# JIWdgClGmrz6dr7HUmaYlNoxpxBxvUnwKGMK9yh8raihRPl78AqIymUztLSwm/vc
|
|
# re9mdMci086xqL406JQ4hu5DOVkAuynICIbPCnHIz19Egx3qOM4Htn1dGe/YeF/x
|
|
# 5N/dHqLTPLiT2XoFfCH6b9jSBElhulUYFd7aQTOAbBUgHUiK2JFuxgfgSwcN7lqx
|
|
# v5SMNX5H626XIFYKc4uyr4f+fuhhl9VPYAx0xMQwJwDBGmuqwqYKj6D3JGuKK2h+
|
|
# TM7O7J5Vfzw9+YeQGDueSCD6hsK3TV5/enxa8v4zpPobj7jcSo+QpGilnUqmEM5V
|
|
# CGZl
|
|
# SIG # End signature block
|
|
|