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.
12559 lines
327 KiB
12559 lines
327 KiB
<#
|
|
.SYNOPSIS
|
|
Sophia Script is a PowerShell module for Windows 10 & Windows 11 fine-tuning and automating the routine tasks
|
|
|
|
Version: v5.12.6
|
|
Date: 23.11.2021
|
|
|
|
Copyright (c) 2014—2021 farag
|
|
Copyright (c) 2019—2021 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 10 versions
|
|
Versions: 2004/20H2/21H1/21H2
|
|
Builds: 19041/19042/19043/19044
|
|
Editions: Home/Pro/Enterprise
|
|
Architecture: x64
|
|
|
|
.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 link
|
|
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 bitness
|
|
switch ([System.Environment]::Is64BitOperatingSystem)
|
|
{
|
|
$false
|
|
{
|
|
Write-Warning -Message $Localization.UnsupportedOSBitness
|
|
exit
|
|
}
|
|
}
|
|
|
|
# Detect the OS build version
|
|
switch (((Get-CimInstance -ClassName Win32_OperatingSystem).BuildNumber -ge 19041) -and ((Get-CimInstance -ClassName Win32_OperatingSystem).BuildNumber -le 19044))
|
|
{
|
|
$false
|
|
{
|
|
Write-Warning -Message $Localization.UnsupportedOSBuild
|
|
exit
|
|
}
|
|
}
|
|
|
|
# Check whether the OS minor build version is 1151 minimum
|
|
switch ((Get-ItemPropertyValue -Path "HKLM:\SOFTWARE\Microsoft\Windows nt\CurrentVersion" -Name UBR) -ge 1151)
|
|
{
|
|
$false
|
|
{
|
|
$Version = Get-ItemPropertyValue -Path "HKLM:\SOFTWARE\Microsoft\Windows nt\CurrentVersion" -Name UBR
|
|
Write-Warning -Message ($Localization.UpdateWarning -f $Version)
|
|
|
|
# Receive 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
|
|
|
|
# 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()
|
|
|
|
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"
|
|
|
|
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
|
|
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)
|
|
exit
|
|
}
|
|
|
|
# Check whether the script was run via PowerShell ISE
|
|
if ($Host.Name -match "ISE")
|
|
{
|
|
Write-Warning -Message $Localization.UnsupportedISE
|
|
exit
|
|
}
|
|
|
|
# Check whether the OS was infected by 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"
|
|
|
|
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 if the current module version is the latest one
|
|
try
|
|
{
|
|
[Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12
|
|
|
|
# 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_10_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"
|
|
exit
|
|
}
|
|
}
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message ($Localization.NoResponse -f "https://github.com/farag2/Sophia-Script-for-Windows")
|
|
Write-Error -Message ($Localization.NoResponse -f "https://github.com/farag2/Sophia-Script-for-Windows") -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
|
|
|
|
# Display a warning message about whether a user has customized the preset file
|
|
if ($Warning)
|
|
{
|
|
$Title = ""
|
|
$Message = $Localization.CustomizationWarning
|
|
$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\..\Sophia.ps1
|
|
|
|
Start-Sleep -Seconds 5
|
|
|
|
Start-Process -FilePath "https://github.com/farag2/Sophia-Script-for-Windows#how-to-use"
|
|
exit
|
|
}
|
|
"1"
|
|
{
|
|
continue
|
|
}
|
|
}
|
|
}
|
|
|
|
# Turn off Controlled folder access to let the script proceed
|
|
switch ((Get-MpPreference).EnableControlledFolderAccess)
|
|
{
|
|
"1"
|
|
{
|
|
Write-Warning -Message $Localization.ControlledFolderAccessDisabled
|
|
|
|
$Script:ControlledFolderAccess = $true
|
|
Set-MpPreference -EnableControlledFolderAccess Disabled
|
|
|
|
# Open "Ransomware protection" page
|
|
Start-Process -FilePath windowsdefender://RansomwareProtection
|
|
}
|
|
"0"
|
|
{
|
|
$Script:ControlledFolderAccess = $false
|
|
}
|
|
}
|
|
|
|
# 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 10" -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
|
|
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
|
|
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"})
|
|
{
|
|
# Security level
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\DataCollection -Name AllowTelemetry -PropertyType DWord -Value 0 -Force
|
|
}
|
|
else
|
|
{
|
|
# Required diagnostic data
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\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
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Policies\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
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.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 10")} | 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 and reopen apps after an update or restart
|
|
|
|
.PARAMETER Disable
|
|
Do not use sign-in info to automatically finish setting up device and reopen apps after an update or restart
|
|
|
|
.PARAMETER Enable
|
|
Use sign-in info to automatically finish setting up device and reopen apps after an update or restart
|
|
|
|
.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 provide locally relevant content by accessing my language list
|
|
|
|
.PARAMETER Enable
|
|
Let websites provide 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 use advertising ID
|
|
|
|
.PARAMETER Disable
|
|
Do not allow apps to use advertising ID to make ads more interresting to you based on your app usage
|
|
|
|
.PARAMETER Enable
|
|
Let apps use advertising ID to make ads more interresting to you based on your app usage
|
|
|
|
.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, trick, and suggestions as you use Windows
|
|
|
|
.PARAMETER Enable
|
|
Get tip, trick, and suggestions as you use Windows
|
|
|
|
.PARAMETER Disable
|
|
Do not get tip, trick, and suggestions as you 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
|
|
Ways I can finish setting up my device to get the most out of Windows
|
|
|
|
.PARAMETER Disable
|
|
Do not suggest ways I can finish setting up my device to get the most out of Windows
|
|
|
|
.PARAMETER Enable
|
|
Suggest ways I can finish setting up my device to get the most out of Windows
|
|
|
|
.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 offer you tailored expereinces based on the diagnostic data setting you have chosen
|
|
|
|
.PARAMETER Enable
|
|
Let Microsoft offer you tailored expereinces based on the diagnostic data setting you have chosen
|
|
|
|
.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 file name extensions
|
|
|
|
.PARAMETER Hide
|
|
Hide 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
|
|
Cortana button on the taskbar
|
|
|
|
.PARAMETER Show
|
|
Hide Cortana button on the taskbar
|
|
|
|
.PARAMETER Hide
|
|
Show Cortana button on the taskbar
|
|
|
|
.EXAMPLE
|
|
CortanaButton -Hide
|
|
|
|
.EXAMPLE
|
|
CortanaButton -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function CortanaButton
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.549981C3F5F10)
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name ShowCortanaButton -PropertyType DWord -Value 0 -Force
|
|
}
|
|
}
|
|
"Show"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.549981C3F5F10)
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name ShowCortanaButton -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
|
|
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 File Explorer ribbon
|
|
|
|
.PARAMETER Expanded
|
|
Expand the File Explorer ribbon
|
|
|
|
.PARAMETER Minimized
|
|
Minimize the File Explorer ribbon
|
|
|
|
.EXAMPLE
|
|
FileExplorerRibbon -Expanded
|
|
|
|
.EXAMPLE
|
|
FileExplorerRibbon -Minimized
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function FileExplorerRibbon
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Expanded"
|
|
)]
|
|
[switch]
|
|
$Expanded,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Minimized"
|
|
)]
|
|
[switch]
|
|
$Minimized
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Expanded"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Ribbon))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Ribbon -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Ribbon -Name MinimizedStateTabletModeOff -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Minimized"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Ribbon))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Ribbon -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Ribbon -Name MinimizedStateTabletModeOff -PropertyType DWord -Value 1 -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
|
|
The "3D Objects" folder in "This PC" and Quick access
|
|
|
|
.PARAMETER Hide
|
|
Hide the "3D Objects" folder in "This PC" and Quick access
|
|
|
|
.PARAMETER Show
|
|
Show the "3D Objects" folder in "This PC" and Quick access
|
|
|
|
.EXAMPLE
|
|
3DObjects -Hide
|
|
|
|
.EXAMPLE
|
|
3DObjects -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function 3DObjects
|
|
{
|
|
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\Explorer\FolderDescriptions\{31C0DD25-9439-4F12-BF41-7FF4EDA38722}\PropertyBag"))
|
|
{
|
|
New-Item -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\FolderDescriptions\{31C0DD25-9439-4F12-BF41-7FF4EDA38722}\PropertyBag" -Force
|
|
}
|
|
New-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\FolderDescriptions\{31C0DD25-9439-4F12-BF41-7FF4EDA38722}\PropertyBag" -Name ThisPCPolicy -PropertyType String -Value Hide -Force
|
|
}
|
|
"Show"
|
|
{
|
|
Remove-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\FolderDescriptions\{31C0DD25-9439-4F12-BF41-7FF4EDA38722}\PropertyBag" -Name ThisPCPolicy -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.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
|
|
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
|
|
People button on the taskbar
|
|
|
|
.PARAMETER Hide
|
|
Hide People button on the taskbar
|
|
|
|
.PARAMETER Show
|
|
Show People button on the taskbar
|
|
|
|
.EXAMPLE
|
|
PeopleTaskbar -Hide
|
|
|
|
.EXAMPLE
|
|
PeopleTaskbar -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function PeopleTaskbar
|
|
{
|
|
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\Microsoft\Windows\CurrentVersion\Explorer\Advanced\People))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced\People -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced\People -Name PeopleBand -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
if (-not (Test-Path -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced\People))
|
|
{
|
|
New-Item -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced\People -Force
|
|
}
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced\People -Name PeopleBand -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Seconds on the taskbar clock
|
|
|
|
.PARAMETER Hide
|
|
Hide seconds on the taskbar clock
|
|
|
|
.PARAMETER Show
|
|
Show seconds on the taskbar clock
|
|
|
|
.EXAMPLE
|
|
SecondsInSystemClock -Hide
|
|
|
|
.EXAMPLE
|
|
SecondsInSystemClock -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function SecondsInSystemClock
|
|
{
|
|
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 ShowSecondsInSystemClock -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\Advanced -Name ShowSecondsInSystemClock -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Search on the taskbar
|
|
|
|
.PARAMETER Hide
|
|
Hide the search on the taskbar
|
|
|
|
.PARAMETER SearchIcon
|
|
Show the search icon on the taskbar
|
|
|
|
.PARAMETER SearchBox
|
|
Show the search box on the taskbar
|
|
|
|
.EXAMPLE
|
|
TaskbarSearch -SearchBox
|
|
|
|
.EXAMPLE
|
|
TaskbarSearch -SearchIcon
|
|
|
|
.EXAMPLE
|
|
TaskbarSearch -Hide
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function TaskbarSearch
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "SearchIcon"
|
|
)]
|
|
[switch]
|
|
$SearchIcon,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "SearchBox"
|
|
)]
|
|
[switch]
|
|
$SearchBox
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Search -Name SearchboxTaskbarMode -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"SearchIcon"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Search -Name SearchboxTaskbarMode -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"SearchBox"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Search -Name SearchboxTaskbarMode -PropertyType DWord -Value 2 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The Windows Ink Workspace button on the taskbar
|
|
|
|
.PARAMETER Hide
|
|
Hide the Windows Ink Workspace button on the taskbar
|
|
|
|
.PARAMETER Show
|
|
Show the Windows Ink Workspace button on the taskbar
|
|
|
|
.EXAMPLE
|
|
WindowsInkWorkspace -Hide
|
|
|
|
.EXAMPLE
|
|
WindowsInkWorkspace -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function WindowsInkWorkspace
|
|
{
|
|
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\PenWorkspace -Name PenWorkspaceButtonDesiredVisibility -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\PenWorkspace -Name PenWorkspaceButtonDesiredVisibility -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Icons in the notification area
|
|
|
|
.PARAMETER Show
|
|
Always show all icons in the notification area
|
|
|
|
.PARAMETER Hide
|
|
Hide all icons in the notification area
|
|
|
|
.EXAMPLE
|
|
NotificationAreaIcons -Show
|
|
|
|
.EXAMPLE
|
|
NotificationAreaIcons -Hide
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function NotificationAreaIcons
|
|
{
|
|
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 -Name EnableAutoTray -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Hide"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer -Name EnableAutoTray -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The Meet Now icon in the notification area
|
|
|
|
.PARAMETER Hide
|
|
Hide the Meet Now icon in the notification area
|
|
|
|
.PARAMETER Show
|
|
Show the Meet Now icon in the notification area
|
|
|
|
.EXAMPLE
|
|
MeetNow -Hide
|
|
|
|
.EXAMPLE
|
|
MeetNow -Show
|
|
|
|
.NOTES
|
|
Current user only
|
|
#>
|
|
function MeetNow
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
$Settings = Get-ItemPropertyValue -Path HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\StuckRects3 -Name Settings -ErrorAction Ignore
|
|
$Settings[9] = 128
|
|
New-ItemProperty -Path HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\StuckRects3 -Name Settings -PropertyType Binary -Value $Settings -Force
|
|
}
|
|
"Show"
|
|
{
|
|
$Settings = Get-ItemPropertyValue -Path HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\StuckRects3 -Name Settings -ErrorAction Ignore
|
|
$Settings[9] = 0
|
|
New-ItemProperty -Path HKCU:\Software\Microsoft\Windows\CurrentVersion\Explorer\StuckRects3 -Name Settings -PropertyType Binary -Value $Settings -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
News and Interests
|
|
|
|
.PARAMETER Disable
|
|
Disable "News and Interests" on the taskbar
|
|
|
|
.PARAMETER Enable
|
|
Enable "News and Interests" on the taskbar
|
|
|
|
.EXAMPLE
|
|
NewsInterests -Disable
|
|
|
|
.EXAMPLE
|
|
NewsInterests -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function NewsInterests
|
|
{
|
|
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\Feeds -Name ShellFeedsTaskbarViewMode -PropertyType DWord -Value 2 -Force
|
|
}
|
|
"Enable"
|
|
{
|
|
New-ItemProperty -Path HKCU:\Software\Microsoft\Windows\CurrentVersion\Feeds -Name ShellFeedsTaskbarViewMode -PropertyType DWord -Value 0 -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
|
|
|
|
.PARAMETER Mail
|
|
Unpin the "Mail" shortcut from the taskbar
|
|
|
|
.EXAMPLE
|
|
UnpinTaskbarShortcuts -Shortcuts Edge, Store, Mail
|
|
|
|
.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", "Mail")]
|
|
[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
|
|
}
|
|
Mail
|
|
{
|
|
# Start-Job is used due to that the calling this function before UninstallUWPApps breaks the retrieval of the localized UWP apps packages names
|
|
if (Get-AppxPackage -Name microsoft.windowscommunicationsapps)
|
|
{
|
|
Start-Job -ScriptBlock {
|
|
$Apps = (New-Object -ComObject Shell.Application).NameSpace("shell:::{4234d49b-0245-4df3-b780-3893943456e1}").Items()
|
|
($Apps | Where-Object -FilterScript {$_.Path -eq "microsoft.windowscommunicationsapps_8wekyb3d8bbwe!microsoft.windowslive.mail"}).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
|
|
The "New App Installed" indicator
|
|
|
|
.PARAMETER Hide
|
|
Hide the "New App Installed" indicator
|
|
|
|
.PARAMETER Show
|
|
Show the "New App Installed" indicator
|
|
|
|
.EXAMPLE
|
|
NewAppInstalledNotification -Hide
|
|
|
|
.EXAMPLE
|
|
NewAppInstalledNotification -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function NewAppInstalledNotification
|
|
{
|
|
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\Policies\Microsoft\Windows\Explorer))
|
|
{
|
|
New-Item -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Force
|
|
}
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name NoNewAppAlert -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
Remove-ItemProperty -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name NoNewAppAlert -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
|
|
)
|
|
|
|
$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
|
|
Show a notification when your PC requires a restart to finish updating
|
|
|
|
.PARAMETER Hide
|
|
Hide a notification when your PC requires a restart 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 %LOCALAPPDATA%\Temp
|
|
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
|
|
{
|
|
# Invoke-Item doesn't work
|
|
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
|
|
|
|
# Restoring closed folders
|
|
foreach ($Script:OpenedFolder in $Script:OpenedFolders)
|
|
{
|
|
if (Test-Path -Path $Script:OpenedFolder)
|
|
{
|
|
Invoke-Item -Path $Script:OpenedFolder
|
|
}
|
|
}
|
|
|
|
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\*\amd64\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)
|
|
{
|
|
# Invoke-Item doesn't work
|
|
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[-1]
|
|
$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
|
|
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
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Clean up of temporary files
|
|
|
|
.PARAMETER Enable
|
|
Delete temporary files that apps aren't using
|
|
|
|
.PARAMETER Disable
|
|
Do not delete temporary files that apps aren't using
|
|
|
|
.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
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#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 $env: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
|
|
The Stop error information display on the BSoD
|
|
|
|
.PARAMETER Enable
|
|
Display the Stop error information on the BSoD
|
|
|
|
.PARAMETER Disable
|
|
Do not display the Stop error information on the BSoD
|
|
|
|
.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 -eq $true)
|
|
{
|
|
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 -eq $true)
|
|
{
|
|
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 10")} | 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*",
|
|
|
|
# Microsoft Paint
|
|
"Microsoft.Windows.MSPaint*",
|
|
|
|
# WordPad
|
|
"Microsoft.Windows.WordPad*"
|
|
)
|
|
|
|
# The following optional features will have their checkboxes unchecked
|
|
[string[]]$UncheckedCapabilities = @(
|
|
# Internet Explorer 11
|
|
"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-RestMethod @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 10")} | 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 when you update Windows
|
|
|
|
.PARAMETER Enable
|
|
Receive updates for other Microsoft products when you update Windows
|
|
|
|
.PARAMETER Disable
|
|
Do not receive updates for other Microsoft products when you update Windows
|
|
|
|
.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 -eq $true)
|
|
{
|
|
(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
|
|
)
|
|
|
|
$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
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Internet Protocol Version 6 (TCP/IPv6) component
|
|
|
|
.PARAMETER Disable
|
|
Disable the Internet Protocol Version 6 (TCP/IPv6) component for all network connections
|
|
|
|
.PARAMETER Enable
|
|
Enable the Internet Protocol Version 6 (TCP/IPv6) component for all network connections
|
|
|
|
.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
|
|
}
|
|
|
|
# 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
|
|
}
|
|
$IPv6Test = Invoke-RestMethod @Parameters | Where-Object -FilterScript {$_.proto -eq "ipv6"}
|
|
}
|
|
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 ($null -eq $IPv6Test)
|
|
{
|
|
Disable-NetAdapterBinding -Name * -ComponentID ms_tcpip6
|
|
}
|
|
}
|
|
"Enable"
|
|
{
|
|
if ($IPv6Test)
|
|
{
|
|
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
|
|
$Change = $Localization.Change
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Change", "&$Skip"
|
|
$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
|
|
$Change = $Localization.Change
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Change", "&$Skip"
|
|
$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
|
|
$Change = $Localization.Change
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Change", "&$Skip"
|
|
$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
|
|
$Change = $Localization.Change
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Change", "&$Skip"
|
|
$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
|
|
$Change = $Localization.Change
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Change", "&$Skip"
|
|
$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
|
|
$Change = $Localization.Change
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Change", "&$Skip"
|
|
$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
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Browse", "&$Skip"
|
|
$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
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Browse", "&$Skip"
|
|
$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
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Browse", "&$Skip"
|
|
$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
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Browse", "&$Skip"
|
|
$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
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Browse", "&$Skip"
|
|
$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
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Browse", "&$Skip"
|
|
$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
|
|
$Change = $Localization.Change
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Change", "&$Skip"
|
|
$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
|
|
$Change = $Localization.Change
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Change", "&$Skip"
|
|
$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
|
|
$Change = $Localization.Change
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Change", "&$Skip"
|
|
$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
|
|
$Change = $Localization.Change
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Change", "&$Skip"
|
|
$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
|
|
$Change = $Localization.Change
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Change", "&$Skip"
|
|
$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
|
|
$Change = $Localization.Change
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Change", "&$Skip"
|
|
$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 OneDrive,
|
|
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"
|
|
{
|
|
if ((Get-Content -Path $PSScriptRoot\..\Sophia.ps1 -Encoding UTF8 -Force | Select-String -SimpleMatch "OneDrive -Uninstall").Line.StartsWith("#") -eq $false)
|
|
{
|
|
$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\Microsoft\Windows\CurrentVersion\Policies\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 keyboard schortcut for Stick keys
|
|
|
|
.PARAMETER Disable
|
|
Turn off pressing the Shift key 5 times to turn Sticky keys
|
|
|
|
.PARAMETER Enable
|
|
Turn on pressing the Shift key 5 times to turn Sticky keys
|
|
|
|
.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 when signing out and restart them after signing in
|
|
|
|
.PARAMETER Disable
|
|
Turn off automatically saving my restartable apps when signing out and restart them after signing 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 ((Get-CimInstance -ClassName CIM_ComputerSystem).PartOfDomain -eq $false)
|
|
{
|
|
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 ((Get-CimInstance -ClassName CIM_ComputerSystem).PartOfDomain -eq $false)
|
|
{
|
|
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 this device as soon as possible when a restart is required to install an update
|
|
|
|
.PARAMETER Enable
|
|
Restart this device as soon as possible when a restart is required to install an update
|
|
|
|
.PARAMETER Disable
|
|
Don't restart this device as soon as possible when a restart is required to install an update
|
|
|
|
.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
|
|
Uninstall the "PC Health Check" app
|
|
|
|
.EXAMPLE
|
|
UninstallPCHealthCheck
|
|
|
|
.LINK
|
|
https://support.microsoft.com/en-us/topic/kb5005463-pc-health-check-application-e33cf4e2-49e2-4727-b913-f3c5b1ee0e56
|
|
|
|
.NOTES
|
|
This application is installed with the KB5005463 update to check if PC meets the system requirements of Windows 11
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function UninstallPCHealthCheck
|
|
{
|
|
$Folder = (New-Object -ComObject Shell.Application).NameSpace("$env:SystemRoot\Installer")
|
|
# Find the necessary .msi with the Subject property equal to "Windows PC Health Check"
|
|
foreach ($MSI in @(Get-ChildItem -Path "$env:SystemRoot\Installer" -Filter *.msi -File -Force))
|
|
{
|
|
$File = $Folder.Items() | Where-Object -FilterScript {$_.Name -eq $MSI.Name}
|
|
# "22" is the "Subject" property for a file
|
|
if ($Folder.GetDetailsOf($File, 22) -eq "Windows PC Health Check")
|
|
{
|
|
Start-Process -FilePath msiexec.exe -ArgumentList "/uninstall $($MSI.FullName) /quiet /norestart" -Wait
|
|
break
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Install the latest supported Microsoft Visual C++ Redistributable 2015—2022 x64
|
|
|
|
.EXAMPLE
|
|
VCRedistx64
|
|
|
|
.LINK
|
|
https://docs.microsoft.com/en-us/cpp/windows/latest-supported-vc-redist
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function InstallVCRedistx64
|
|
{
|
|
$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/16/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
|
|
|
|
Remove-Item -Path "$DownloadsFolder\vc_redist.x64.exe", "$env:TEMP\dd_vcredist_amd64_*.log" -Force -ErrorAction Ignore
|
|
}
|
|
#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
|
|
To receive kernel updates, enable the Windows Update setting: "Receive updates for other Microsoft products"
|
|
|
|
.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 ($ButtonInstall.IsEnabled -eq $false)
|
|
{
|
|
$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 10")} | 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
|
|
}
|
|
#endregion WSL
|
|
|
|
#region Start menu
|
|
<#
|
|
.SYNOPSIS
|
|
Recently added apps in the Start menu
|
|
|
|
.PARAMETER Hide
|
|
Hide recently added apps in the Start menu
|
|
|
|
.PARAMETER Show
|
|
Show recently added apps in the Start menu
|
|
|
|
.EXAMPLE
|
|
RecentlyAddedApps -Hide
|
|
|
|
.EXAMPLE
|
|
RecentlyAddedApps -Show
|
|
|
|
.NOTES
|
|
Machine-wide
|
|
#>
|
|
function RecentlyAddedApps
|
|
{
|
|
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\Policies\Microsoft\Windows\Explorer))
|
|
{
|
|
New-Item -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Force
|
|
}
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name HideRecentlyAddedApps -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
Remove-ItemProperty -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name HideRecentlyAddedApps -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
App suggestions in the Start menu
|
|
|
|
.PARAMETER Hide
|
|
Hide app suggestions in the Start menu
|
|
|
|
.PARAMETER Show
|
|
Show app suggestions in the Start menu
|
|
|
|
.EXAMPLE
|
|
AppSuggestions -Hide
|
|
|
|
.EXAMPLE
|
|
AppSuggestions -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function AppSuggestions
|
|
{
|
|
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-338388Enabled -PropertyType DWord -Value 0 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\ContentDeliveryManager -Name SubscribedContent-338388Enabled -PropertyType DWord -Value 1 -Force
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.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
|
|
The Start tiles
|
|
|
|
.PARAMETER ControlPanel
|
|
Pin the "Control Panel" shortcut to Start
|
|
|
|
.PARAMETER DevicesPrinters
|
|
Pin the "Devices & Printers" shortcut to Start
|
|
|
|
.PARAMETER PowerShell
|
|
Pin the "Windows PowerShell" shortcut to Start
|
|
|
|
.PARAMETER UnpinAll
|
|
Unpin all the Start tiles
|
|
|
|
.EXAMPLE
|
|
PinToStart -Tiles ControlPanel, DevicesPrinters, PowerShell
|
|
|
|
.EXAMPLE
|
|
PinToStart -UnpinAll
|
|
|
|
.EXAMPLE
|
|
PinToStart -UnpinAll -Tiles ControlPanel, DevicesPrinters, PowerShell
|
|
|
|
.EXAMPLE
|
|
PinToStart -UnpinAll -Tiles ControlPanel
|
|
|
|
.EXAMPLE
|
|
PinToStart -Tiles ControlPanel -UnpinAll
|
|
|
|
.NOTES
|
|
Separate arguments with a comma
|
|
Current user
|
|
#>
|
|
function PinToStart
|
|
{
|
|
[CmdletBinding()]
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $false,
|
|
Position = 0
|
|
)]
|
|
[switch]
|
|
$UnpinAll,
|
|
|
|
[Parameter(
|
|
Mandatory = $false,
|
|
Position = 1
|
|
)]
|
|
[ValidateSet("ControlPanel", "DevicesPrinters", "PowerShell")]
|
|
[string[]]
|
|
$Tiles
|
|
)
|
|
|
|
begin
|
|
{
|
|
$Script:StartLayout = "$PSScriptRoot\..\StartLayout.xml"
|
|
|
|
# Unpin all the Start tiles
|
|
if ($UnpinAll)
|
|
{
|
|
# Export the current Start layout
|
|
Export-StartLayout -Path $Script:StartLayout -UseDesktopApplicationID
|
|
|
|
[xml]$XML = Get-Content -Path $Script:StartLayout -Encoding UTF8 -Force
|
|
$Groups = $XML.LayoutModificationTemplate.DefaultLayoutOverride.StartLayoutCollection.StartLayout.Group
|
|
|
|
foreach ($Group in $Groups)
|
|
{
|
|
# Removing all groups inside XML
|
|
$Group.ParentNode.RemoveChild($Group) | Out-Null
|
|
}
|
|
|
|
$XML.Save($Script:StartLayout)
|
|
}
|
|
}
|
|
|
|
process
|
|
{
|
|
# Extract strings from shell32.dll using its' number
|
|
# https://github.com/Disassembler0/Win10-Initial-Setup-Script/issues/8#issue-227159084
|
|
$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 "Devices and Printers" string from shell32.dll
|
|
$DevicesPrinters = [WinAPI.GetStr]::GetString(30493)
|
|
|
|
# Check if an argument is "DevicesPrinters". The Devices and Printers's AppID attribute can be retrieved only if the shortcut was created
|
|
if (((Get-Command -Name PinToStart).Parametersets.Parameters | Where-Object -FilterScript {$null -eq $_.Attributes.AliasNames}).Attributes.ValidValues | Where-Object -FilterScript {$_ -match "DevicesPrinters"})
|
|
{
|
|
# Create the old-style "Devices and Printers" shortcut in the Start menu
|
|
$Shell = New-Object -ComObject Wscript.Shell
|
|
$Shortcut = $Shell.CreateShortcut("$env:APPDATA\Microsoft\Windows\Start menu\Programs\System Tools\$DevicesPrinters.lnk")
|
|
$Shortcut.TargetPath = "control"
|
|
$Shortcut.Arguments = "printers"
|
|
$Shortcut.IconLocation = "$env:SystemRoot\system32\DeviceCenter.dll"
|
|
$Shortcut.Save()
|
|
|
|
Start-Sleep -Seconds 3
|
|
}
|
|
|
|
# Get the AppID because it's auto generated AppID for the "Devices and Printers" shortcut
|
|
$DevicesPrintersAppID = (Get-StartApps | Where-Object -FilterScript {$_.Name -eq $DevicesPrinters}).AppID
|
|
|
|
$Parameters = @(
|
|
# Control Panel hash table
|
|
@{
|
|
# Special name for Control Panel
|
|
Name = "ControlPanel"
|
|
Size = "2x2"
|
|
Column = 0
|
|
Row = 0
|
|
AppID = "Microsoft.Windows.ControlPanel"
|
|
},
|
|
# "Devices & Printers" hash table
|
|
@{
|
|
# Special name for "Devices & Printers"
|
|
Name = "DevicesPrinters"
|
|
Size = "2x2"
|
|
Column = 2
|
|
Row = 0
|
|
AppID = $DevicesPrintersAppID
|
|
},
|
|
# Windows PowerShell hash table
|
|
@{
|
|
# Special name for Windows PowerShell
|
|
Name = "PowerShell"
|
|
Size = "2x2"
|
|
Column = 4
|
|
Row = 0
|
|
AppID = "{1AC14E77-02E7-4E5D-B744-2EB1AE5198B7}\WindowsPowerShell\v1.0\powershell.exe"
|
|
}
|
|
)
|
|
|
|
# Valid columns to place tiles in
|
|
$ValidColumns = @(0, 2, 4)
|
|
[string]$StartLayoutNS = "http://schemas.microsoft.com/Start/2014/StartLayout"
|
|
|
|
# Add pre-configured hastable to XML
|
|
function Add-Tile
|
|
{
|
|
param
|
|
(
|
|
[string]
|
|
$Size,
|
|
|
|
[int]
|
|
$Column,
|
|
|
|
[int]
|
|
$Row,
|
|
|
|
[string]
|
|
$AppID
|
|
)
|
|
|
|
[string]$elementName = "start:DesktopApplicationTile"
|
|
[Xml.XmlElement]$Table = $xml.CreateElement($elementName, $StartLayoutNS)
|
|
$Table.SetAttribute("Size", $Size)
|
|
$Table.SetAttribute("Column", $Column)
|
|
$Table.SetAttribute("Row", $Row)
|
|
$Table.SetAttribute("DesktopApplicationID", $AppID)
|
|
|
|
$Table
|
|
}
|
|
|
|
if (-not (Test-Path -Path $Script:StartLayout))
|
|
{
|
|
# Export the current Start layout
|
|
Export-StartLayout -Path $Script:StartLayout -UseDesktopApplicationID
|
|
}
|
|
|
|
[xml]$XML = Get-Content -Path $Script:StartLayout -Encoding UTF8 -Force
|
|
|
|
foreach ($Tile in $Tiles)
|
|
{
|
|
switch ($Tile)
|
|
{
|
|
ControlPanel
|
|
{
|
|
$ControlPanel = [WinAPI.GetStr]::GetString(12712)
|
|
Write-Verbose -Message ($Localization.ShortcutPinning -f $ControlPanel) -Verbose
|
|
}
|
|
DevicesPrinters
|
|
{
|
|
Write-Verbose -Message ($Localization.ShortcutPinning -f $DevicesPrinters) -Verbose
|
|
}
|
|
PowerShell
|
|
{
|
|
Write-Verbose -Message ($Localization.ShortcutPinning -f "Windows PowerShell") -Verbose
|
|
}
|
|
}
|
|
|
|
$Parameter = $Parameters | Where-Object -FilterScript {$_.Name -eq $Tile}
|
|
$Group = $XML.LayoutModificationTemplate.DefaultLayoutOverride.StartLayoutCollection.StartLayout.Group | Where-Object -FilterScript {$_.Name -eq "Sophia Script"}
|
|
|
|
# If the "Sophia Script" group exists in Start
|
|
if ($Group)
|
|
{
|
|
$DesktopApplicationID = ($Parameters | Where-Object -FilterScript {$_.Name -eq $Tile}).AppID
|
|
|
|
if (-not ($Group.DesktopApplicationTile | Where-Object -FilterScript {$_.DesktopApplicationID -eq $DesktopApplicationID}))
|
|
{
|
|
# Calculate current filled columns
|
|
$CurrentColumns = @($Group.DesktopApplicationTile.Column)
|
|
# Calculate current free columns and take the first one
|
|
$Column = (Compare-Object -ReferenceObject $ValidColumns -DifferenceObject $CurrentColumns).InputObject | Select-Object -First 1
|
|
# If filled cells contain desired ones assign the first free column
|
|
if ($CurrentColumns -contains $Parameter.Column)
|
|
{
|
|
$Parameter.Column = $Column
|
|
}
|
|
$Group.AppendChild((Add-Tile @Parameter)) | Out-Null
|
|
}
|
|
}
|
|
else
|
|
{
|
|
# Create the "Sophia Script" group
|
|
[Xml.XmlElement]$Group = $XML.CreateElement("start:Group", $StartLayoutNS)
|
|
$Group.SetAttribute("Name","Sophia Script")
|
|
$Group.AppendChild((Add-Tile @Parameter)) | Out-Null
|
|
$XML.LayoutModificationTemplate.DefaultLayoutOverride.StartLayoutCollection.StartLayout.AppendChild($Group) | Out-Null
|
|
}
|
|
}
|
|
|
|
$XML.Save($Script:StartLayout)
|
|
}
|
|
|
|
end
|
|
{
|
|
# Temporarily disable changing the Start menu layout
|
|
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 LockedStartLayout -PropertyType DWord -Value 1 -Force
|
|
New-ItemProperty -Path HKCU:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name StartLayoutFile -PropertyType ExpandString -Value $Script:StartLayout -Force
|
|
|
|
Start-Sleep -Seconds 3
|
|
|
|
# Restart the Start menu
|
|
Stop-Process -Name StartMenuExperienceHost -Force -ErrorAction Ignore
|
|
|
|
Start-Sleep -Seconds 3
|
|
|
|
# Open the Start menu to load the new layout
|
|
$wshell = New-Object -ComObject WScript.Shell
|
|
$wshell.SendKeys("^{ESC}")
|
|
|
|
Start-Sleep -Seconds 3
|
|
|
|
# Enable changing the Start menu layout
|
|
Remove-ItemProperty -Path HKCU:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name LockedStartLayout -Force -ErrorAction Ignore
|
|
Remove-ItemProperty -Path HKCU:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name StartLayoutFile -Force -ErrorAction Ignore
|
|
|
|
Remove-Item -Path $Script:StartLayout -Force
|
|
|
|
Stop-Process -Name StartMenuExperienceHost -Force -ErrorAction Ignore
|
|
|
|
Start-Sleep -Seconds 3
|
|
|
|
# Open the Start menu to load the new layout
|
|
$wshell = New-Object -ComObject WScript.Shell
|
|
$wshell.SendKeys("^{ESC}")
|
|
}
|
|
}
|
|
#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 UWP panel
|
|
"AdvancedMicroDevicesInc*",
|
|
|
|
# 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",
|
|
|
|
# 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",
|
|
|
|
# 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",
|
|
|
|
# Microsoft Store
|
|
"Microsoft.WindowsStore",
|
|
|
|
# Windows Terminal
|
|
"Microsoft.WindowsTerminal",
|
|
"Microsoft.WindowsTerminalPreview",
|
|
|
|
# Web Media Extensions
|
|
"Microsoft.WebMediaExtensions"
|
|
)
|
|
|
|
#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}
|
|
$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 10")} | 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}
|
|
$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 10")} | 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 using the https://store.rg-adguard.net parser
|
|
|
|
.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
|
|
|
|
.LINK
|
|
https://dev.to/kaiwalter/download-windows-store-apps-with-powershell-from-https-store-rg-adguard-net-155m
|
|
|
|
.NOTES
|
|
The extension can be installed without Microsoft account
|
|
|
|
.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 ((-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
|
|
}
|
|
|
|
try
|
|
{
|
|
# Check whether the https://store.rg-adguard.net site is alive
|
|
$Parameters = @{
|
|
Uri = "https://store.rg-adguard.net/api/GetFiles"
|
|
Method = "Head"
|
|
DisableKeepAlive = $true
|
|
UseBasicParsing = $true
|
|
}
|
|
if (-not (Invoke-RestMethod @Parameters).StatusDescription)
|
|
{
|
|
return
|
|
}
|
|
|
|
$Parameters = @{
|
|
Method = "Post"
|
|
Uri = "https://store.rg-adguard.net/api/GetFiles"
|
|
ContentType = "application/x-www-form-urlencoded"
|
|
Body = @{
|
|
type = "url"
|
|
# HEVC Video Extensions from Device Manufacturer
|
|
url = "https://www.microsoft.com/store/productId/9n4wgh0z6vhq"
|
|
ring = "Retail"
|
|
lang = "en-US"
|
|
}
|
|
UseBasicParsing = $true
|
|
}
|
|
$Raw = Invoke-WebRequest @Parameters
|
|
|
|
# Parsing the page
|
|
$Raw | Select-String -Pattern '<tr style.*<a href=\"(?<url>.*)"\s.*>(?<text>.*)<\/a>' -AllMatches | ForEach-Object -Process {$_.Matches} | ForEach-Object -Process {
|
|
$TempURL = $_.Groups[1].Value
|
|
$Package = $_.Groups[2].Value
|
|
|
|
if ($Package -like "Microsoft.HEVCVideoExtension_*_x64__8wekyb3d8bbwe.appx")
|
|
{
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.HEVCDownloading -Verbose
|
|
|
|
$DownloadsFolder = Get-ItemPropertyValue -Path "HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders" -Name "{374DE290-123F-4565-9164-39C4925E467B}"
|
|
$Parameters = @{
|
|
Uri = $TempURL
|
|
OutFile = "$DownloadsFolder\$Package"
|
|
UseBasicParsing = $true
|
|
Verbose = $true
|
|
}
|
|
Invoke-WebRequest @Parameters
|
|
|
|
# Installing "HEVC Video Extensions from Device Manufacturer"
|
|
Add-AppxPackage -Path "$DownloadsFolder\$Package" -Verbose
|
|
|
|
Remove-Item -Path "$DownloadsFolder\$Package" -Force
|
|
}
|
|
}
|
|
}
|
|
catch [System.Net.WebException]
|
|
{
|
|
Write-Warning -Message ($Localization.NoResponse -f "https://store.rg-adguard.net")
|
|
Write-Error -Message ($Localization.NoResponse -f "https://store.rg-adguard.net") -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
|
|
How UWP runs in the background
|
|
|
|
.PARAMETER Disable
|
|
Do not let UWP apps run in the background
|
|
|
|
.PARAMETER Enable
|
|
Let all UWP apps run in the background
|
|
|
|
.EXAMPLE
|
|
BackgroundUWPApps -Disable
|
|
|
|
.EXAMPLE
|
|
BackgroundUWPApps -Enable
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function BackgroundUWPApps
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Disable"
|
|
{
|
|
# Remove all excluded apps running in the background
|
|
Get-ChildItem -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\BackgroundAccessApplications | ForEach-Object -Process {
|
|
Remove-ItemProperty -Path $_.PsPath -Name * -Force
|
|
}
|
|
|
|
# Exclude apps from the Bundle only
|
|
$BackgroundAccessApplications = @((Get-ChildItem -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\BackgroundAccessApplications).PSChildName)
|
|
$ExcludedBackgroundAccessApplications = @()
|
|
foreach ($BackgroundAccessApplication in $BackgroundAccessApplications)
|
|
{
|
|
if (Get-AppxPackage -PackageTypeFilter Bundle -AllUsers | Where-Object -FilterScript {$_.PackageFamilyName -eq $BackgroundAccessApplication})
|
|
{
|
|
$ExcludedBackgroundAccessApplications += $BackgroundAccessApplication
|
|
}
|
|
}
|
|
|
|
Get-ChildItem -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\BackgroundAccessApplications | Where-Object -FilterScript {$_.PSChildName -in $ExcludedBackgroundAccessApplications} | ForEach-Object -Process {
|
|
New-ItemProperty -Path $_.PsPath -Name Disabled -PropertyType DWord -Value 1 -Force
|
|
New-ItemProperty -Path $_.PsPath -Name DisabledByUser -PropertyType DWord -Value 1 -Force
|
|
}
|
|
|
|
# Open the "Background apps" page
|
|
Start-Process -FilePath ms-settings:privacy-backgroundapps
|
|
}
|
|
"Enable"
|
|
{
|
|
Get-ChildItem -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\BackgroundAccessApplications | ForEach-Object -Process {
|
|
Remove-ItemProperty -Path $_.PsPath -Name * -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
|
|
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
|
|
$Add = $Localization.Add
|
|
$Skip = $Localization.Skip
|
|
$Options = "&$Add", "&$Skip"
|
|
$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"
|
|
{
|
|
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
|
|
"Delivery Optimization Files",
|
|
|
|
# Device driver packages
|
|
"Device Driver Packages",
|
|
|
|
# Previous Windows Installation(s)
|
|
"Previous Installations",
|
|
|
|
# Setup log files
|
|
"Setup Log Files",
|
|
|
|
# Temporary Setup Files
|
|
"Temporary Setup Files",
|
|
|
|
# Windows Update Cleanup
|
|
"Update Cleanup",
|
|
|
|
# Microsoft Defender
|
|
"Windows Defender",
|
|
|
|
# Windows upgrade log files
|
|
"Windows Upgrade Log Files"
|
|
)
|
|
foreach ($VolumeCache in $VolumeCaches)
|
|
{
|
|
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 -TaskName "Windows Cleanup" -Confirm:$false -ErrorAction Ignore
|
|
Unregister-ScheduledTask -TaskName "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"
|
|
{
|
|
# 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 -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"
|
|
{
|
|
$TempTask = @"
|
|
Get-ChildItem -Path `$env:TEMP -Recurse -Force | Where-Object {`$_.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 -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 ((Get-MpComputerStatus).AntivirusEnabled -eq $true)
|
|
{
|
|
Set-MpPreference -EnableNetworkProtection Enabled
|
|
}
|
|
}
|
|
"Disable"
|
|
{
|
|
if ((Get-MpComputerStatus).AntivirusEnabled -eq $true)
|
|
{
|
|
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 ((Get-MpComputerStatus).AntivirusEnabled -eq $true)
|
|
{
|
|
Set-MpPreference -PUAProtection Enabled
|
|
}
|
|
}
|
|
"Disable"
|
|
{
|
|
if ((Get-MpComputerStatus).AntivirusEnabled -eq $true)
|
|
{
|
|
Set-MpPreference -PUAProtection Disabled
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
Sandboxing for Microsoft Defender
|
|
|
|
.PARAMETER Enable
|
|
Enable sandboxing for Microsoft Defender
|
|
|
|
.PARAMETER Disable
|
|
Disable sandboxing for Microsoft Defender
|
|
|
|
.EXAMPLE
|
|
DefenderSandbox -Enable
|
|
|
|
.EXAMPLE
|
|
DefenderSandbox -Disable
|
|
|
|
.NOTES
|
|
There is a bug in KVM with QEMU: enabling this function causes VM to freeze up during the loading phase of Windows
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function DefenderSandbox
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Enable"
|
|
)]
|
|
[switch]
|
|
$Enable,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Disable"
|
|
)]
|
|
[switch]
|
|
$Disable
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Enable"
|
|
{
|
|
if ((Get-MpComputerStatus).AntivirusEnabled -eq $true)
|
|
{
|
|
setx /M MP_FORCE_USE_SANDBOX 1
|
|
}
|
|
}
|
|
"Disable"
|
|
{
|
|
if ((Get-MpComputerStatus).AntivirusEnabled -eq $true)
|
|
{
|
|
setx /M MP_FORCE_USE_SANDBOX 0
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
# 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" Event Viewer сustom view to log the executed processes and their arguments
|
|
|
|
.PARAMETER Disable
|
|
Remove the "Process Creation" Event Viewer custom view
|
|
|
|
.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 -eq $true)
|
|
{
|
|
Disable-WindowsOptionalFeature -FeatureName Containers-DisposableClientVM -Online -NoRestart
|
|
}
|
|
else
|
|
{
|
|
try
|
|
{
|
|
# Determining whether Hyper-V is enabled
|
|
if ((Get-CimInstance -ClassName CIM_ComputerSystem).HypervisorPresent -eq $true)
|
|
{
|
|
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 -eq $true)
|
|
{
|
|
Enable-WindowsOptionalFeature -FeatureName Containers-DisposableClientVM -All -Online -NoRestart
|
|
}
|
|
else
|
|
{
|
|
try
|
|
{
|
|
# Determining whether Hyper-V is enabled
|
|
if ((Get-CimInstance -ClassName CIM_ComputerSystem).HypervisorPresent -eq $true)
|
|
{
|
|
Enable-WindowsOptionalFeature -FeatureName Containers-DisposableClientVM -All -Online -NoRestart
|
|
}
|
|
}
|
|
catch [System.Exception]
|
|
{
|
|
Write-Error -Message $Localization.EnableHardwareVT -ErrorAction SilentlyContinue
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
#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"
|
|
{
|
|
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 "{E2BF9676-5F8F-435C-97EB-11607A5BEDF7}" -PropertyType String -Value "" -Force
|
|
}
|
|
"Show"
|
|
{
|
|
Remove-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows\CurrentVersion\Shell Extensions\Blocked" -Name "{E2BF9676-5F8F-435C-97EB-11607A5BEDF7}" -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.SYNOPSIS
|
|
The "Edit with Paint 3D" item in the media files context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Edit with Paint 3D" item from the media files context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Edit with Paint 3D" item in the media files context menu
|
|
|
|
.EXAMPLE
|
|
EditWithPaint3DContext -Hide
|
|
|
|
.EXAMPLE
|
|
EditWithPaint3DContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function EditWithPaint3DContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.MSPaint)
|
|
{
|
|
$Extensions = @(".bmp", ".gif", ".jpe", ".jpeg", ".jpg", ".png", ".tif", ".tiff")
|
|
foreach ($Extension in $Extensions)
|
|
{
|
|
New-ItemProperty -Path "Registry::HKEY_CLASSES_ROOT\SystemFileAssociations\$Extension\Shell\3D Edit" -Name ProgrammaticAccessOnly -PropertyType String -Value "" -Force
|
|
}
|
|
}
|
|
}
|
|
"Show"
|
|
{
|
|
if (Get-AppxPackage -Name Microsoft.MSPaint)
|
|
{
|
|
$Extensions = @(".bmp", ".gif", ".jpe", ".jpeg", ".jpg", ".png", ".tif", ".tiff")
|
|
foreach ($Extension in $Extensions)
|
|
{
|
|
Remove-ItemProperty -Path "Registry::HKEY_CLASSES_ROOT\SystemFileAssociations\$Extension\Shell\3D Edit" -Name ProgrammaticAccessOnly -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
<#
|
|
.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 "Edit" item in the images context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Edit" item from the images context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Edit" item in the images context menu
|
|
|
|
.EXAMPLE
|
|
ImagesEditContext -Hide
|
|
|
|
.EXAMPLE
|
|
ImagesEditContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function ImagesEditContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
if ((Get-WindowsCapability -Online -Name "Microsoft.Windows.MSPaint*").State -eq "Installed")
|
|
{
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\SystemFileAssociations\image\shell\edit -Name ProgrammaticAccessOnly -PropertyType String -Value "" -Force
|
|
}
|
|
}
|
|
"Show"
|
|
{
|
|
if ((Get-WindowsCapability -Online -Name "Microsoft.Windows.MSPaint*").State -eq "Installed")
|
|
{
|
|
Remove-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\SystemFileAssociations\image\shell\edit -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 "Bitmap image" item in the "New" context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Bitmap image" item from the "New" context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Bitmap image" item to the "New" context menu
|
|
|
|
.EXAMPLE
|
|
BitmapImageNewContext -Hide
|
|
|
|
.EXAMPLE
|
|
BitmapImageNewContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function BitmapImageNewContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
if ((Get-WindowsCapability -Online -Name "Microsoft.Windows.MSPaint*").State -eq "Installed")
|
|
{
|
|
Remove-Item -Path Registry::HKEY_CLASSES_ROOT\.bmp\ShellNew -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
"Show"
|
|
{
|
|
if ((Get-WindowsCapability -Online -Name "Microsoft.Windows.MSPaint*").State -eq "Installed")
|
|
{
|
|
if (-not (Test-Path -Path Registry::HKEY_CLASSES_ROOT\.bmp\ShellNew))
|
|
{
|
|
New-Item -Path Registry::HKEY_CLASSES_ROOT\.bmp\ShellNew -Force
|
|
}
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\.bmp\ShellNew -Name ItemName -PropertyType ExpandString -Value "@%systemroot%\system32\mspaint.exe,-59414" -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\.bmp\ShellNew -Name NullFile -PropertyType String -Value "" -Force
|
|
}
|
|
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
|
|
}
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
Get-WindowsCapability -Online -Name "Microsoft.Windows.MSPaint*" | Add-WindowsCapability -Online
|
|
}
|
|
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
|
|
The "Rich Text Document" item in the "New" context menu
|
|
|
|
.PARAMETER Hide
|
|
Hide the "Rich Text Document" item from the "New" context menu
|
|
|
|
.PARAMETER Show
|
|
Show the "Rich Text Document" item to the "New" context menu
|
|
|
|
.EXAMPLE
|
|
RichTextDocumentNewContext -Hide
|
|
|
|
.EXAMPLE
|
|
RichTextDocumentNewContext -Show
|
|
|
|
.NOTES
|
|
Current user
|
|
#>
|
|
function RichTextDocumentNewContext
|
|
{
|
|
param
|
|
(
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Hide"
|
|
)]
|
|
[switch]
|
|
$Hide,
|
|
|
|
[Parameter(
|
|
Mandatory = $true,
|
|
ParameterSetName = "Show"
|
|
)]
|
|
[switch]
|
|
$Show
|
|
)
|
|
|
|
switch ($PSCmdlet.ParameterSetName)
|
|
{
|
|
"Hide"
|
|
{
|
|
if ((Get-WindowsCapability -Online -Name "Microsoft.Windows.WordPad*").State -eq "Installed")
|
|
{
|
|
Remove-Item -Path Registry::HKEY_CLASSES_ROOT\.rtf\ShellNew -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
"Show"
|
|
{
|
|
if ((Get-WindowsCapability -Online -Name "Microsoft.Windows.WordPad*").State -eq "Installed")
|
|
{
|
|
if (-not (Test-Path -Path Registry::HKEY_CLASSES_ROOT\.rtf\ShellNew))
|
|
{
|
|
New-Item -Path Registry::HKEY_CLASSES_ROOT\.rtf\ShellNew -Force
|
|
}
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\.rtf\ShellNew -Name Data -PropertyType String -Value "{\rtf1}" -Force
|
|
New-ItemProperty -Path Registry::HKEY_CLASSES_ROOT\.rtf\ShellNew -Name ItemName -PropertyType ExpandString -Value "@%ProgramFiles%\Windows NT\Accessories\WORDPAD.EXE,-213" -Force
|
|
}
|
|
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
|
|
}
|
|
|
|
Write-Information -MessageData "" -InformationAction Continue
|
|
Write-Verbose -Message $Localization.Patient -Verbose
|
|
|
|
Get-WindowsCapability -Online -Name "Microsoft.Windows.WordPad*" | Add-WindowsCapability -Online
|
|
}
|
|
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
|
|
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 HKLM:\SOFTWARE\Policies\Microsoft\Windows\Explorer))
|
|
{
|
|
New-Item -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Force
|
|
}
|
|
New-ItemProperty -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name NoUseStoreOpenWith -PropertyType DWord -Value 1 -Force
|
|
}
|
|
"Show"
|
|
{
|
|
Remove-ItemProperty -Path HKLM:\SOFTWARE\Policies\Microsoft\Windows\Explorer -Name NoUseStoreOpenWith -Force -ErrorAction Ignore
|
|
}
|
|
}
|
|
}
|
|
#endregion Context menu
|
|
|
|
#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:ControlledFolderAccess)
|
|
{
|
|
Set-MpPreference -EnableControlledFolderAccess Enabled
|
|
}
|
|
|
|
# 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)
|
|
{
|
|
Invoke-Item -Path $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()
|
|
}
|
|
}
|
|
|