Remove-MS-Store-Apps.ps1
<#PSScriptInfo
.VERSION 24.11.09 .GUID 888f5987-8b64-4a4a-ab8e-00a1bc99ff54 .AUTHOR Mike Galvin Contact: digressive@outlook.com .COMPANYNAME Mike Galvin .COPYRIGHT (C) Mike Galvin. All rights reserved. .TAGS Remove Clean up Microsoft Store Windows UWP in-box built-in included app Windows 11 10 Customisable removal utility .LICENSEURI https://github.com/Digressive/Remove-MS-Store-Apps?tab=MIT-1-ov-file .PROJECTURI https://gal.vin/utils/remove-ms-store-apps-utility/ .ICONURI .EXTERNALMODULEDEPENDENCIES .REQUIREDSCRIPTS .EXTERNALSCRIPTDEPENDENCIES .RELEASENOTES #> <# .SYNOPSIS Remove MS Store Apps - Customisable Windows 11/10 Microsoft Store app removal utility .DESCRIPTION Removes apps included in Windows 11/10. Run with -help or no arguments for usage. #> ## Set up command line switches. [CmdletBinding()] Param( [alias("List")] $AppListFile, [alias("Wim")] $WimFile, [alias("WimIndex")] $WIndex, [alias("WimMountPath")] $WimMntPath, [alias("L")] $LogPathUsr, [alias("LogRotate")] $LogHistory, [switch]$PCApps, [switch]$UserApps, [switch]$Uno, [switch]$PCOnly, [switch]$Help, [switch]$NoBanner) If ($NoBanner -eq $False) { Write-Host -ForegroundColor Yellow -BackgroundColor Black -Object " _____ __ __ _____ _____ _ | __ \ | \/ |/ ____| / ____| | | |__) |___ _ __ ___ _____ _____ | \ / | (___ | (___ | |_ ___ _ __ ___ | _ // _ \ '_ ' _ \ / _ \ \ / / _ \ | |\/| |\___ \ \___ \| __/ _ \| '__/ _ \ | | \ \ __/ | | | | | (_) \ V / __/ | | | |____) | ____) | || (_) | | | __/ |_| \_\___|_| |_| |_|\___/ \_/ \___| |_|_ |_|_____/ |_____/ \__\___/|_| \___| /\ | | | | | (_) (_) | / \ _ __ _ __ ___ | | | | |_ _| |_| |_ _ _ / /\ \ | '_ \| '_ \/ __| | | | | __| | | | __| | | | Mike Galvin / ____ \| |_) | |_) \__ \ | |__| | |_| | | | |_| |_| | https://gal.vin /_/ \_\ .__/| .__/|___/ \____/ \__|_|_|_|\__|\__, | | | | | __/ | Version 24.11.09 |_| |_| |___/ See -help for usage Donate: https://www.paypal.me/digressive " } If ($PSBoundParameters.Values.Count -eq 0 -or $Help) { Write-Host -Object "Usage: From an elevated terminal run: [path\]Remove-MS-Store-Apps.ps1 -List [path\]apps-to-remove.txt This will remove the apps in the txt file from your Windows installation for all users. You can also run: [path\]Remove-MS-Store-Apps.ps1 -Uno -List [path\]apps-to-keep.txt This will remove the all apps not in the txt file from your Windows installation for all users. To remove only PC apps and leave the current user apps untouched use: -PCOnly To list apps for all users: -PCApps. To list apps for the current user: -UserApps. To operate on a wim file: -Wim [path\]install.wim -WimIndex [number] (optional: -WimMountPath [path\]) Run the following command to find out the WimIndex for your wim file: Get-WindowsImage -ImagePath [path\]install.wim | Format-Table -Property ImageIndex, ImageName To output a log: -L [path\]. To remove logs produced by the utility older than X days: -LogRotate [number]. Run with no ASCII banner: -NoBanner" } else { ## If logging is configured, start logging. ## If the log file already exists, clear it. If ($LogPathUsr) { ## Clean User entered string $LogPath = $LogPathUsr.trimend('\') ## Make sure the log directory exists. If ((Test-Path -Path $LogPath) -eq $False) { New-Item $LogPath -ItemType Directory -Force | Out-Null } $LogFile = ("Remove-MS-Store-Apps_{0:yyyy-MM-dd_HH-mm-ss}.log" -f (Get-Date)) $Log = "$LogPath\$LogFile" If (Test-Path -Path $Log) { Clear-Content -Path $Log } } ## Function to get date in specific format. Function Get-DateFormat { Get-Date -Format "yyyy-MM-dd HH:mm:ss" } ## Function for logging. Function Write-Log($Type, $Evt) { If ($Type -eq "Info") { If ($LogPathUsr) { Add-Content -Path $Log -Encoding ASCII -Value "$(Get-DateFormat) [INFO] $Evt" } Write-Host -Object "$(Get-DateFormat) [INFO] $Evt" } If ($Type -eq "Succ") { If ($LogPathUsr) { Add-Content -Path $Log -Encoding ASCII -Value "$(Get-DateFormat) [SUCCESS] $Evt" } Write-Host -ForegroundColor Green -Object "$(Get-DateFormat) [SUCCESS] $Evt" } If ($Type -eq "Err") { If ($LogPathUsr) { Add-Content -Path $Log -Encoding ASCII -Value "$(Get-DateFormat) [ERROR] $Evt" } Write-Host -ForegroundColor Red -BackgroundColor Black -Object "$(Get-DateFormat) [ERROR] $Evt" } If ($Type -eq "Conf") { If ($LogPathUsr) { Add-Content -Path $Log -Encoding ASCII -Value "$Evt" } Write-Host -ForegroundColor Cyan -Object "$Evt" } } ## Function for Update Check Function UpdateCheck() { $ScriptVersion = "24.11.09" $RawSource = "https://raw.githubusercontent.com/Digressive/Remove-MS-Store-Apps/master/Remove-MS-Store-Apps.ps1" try { $SourceCheck = Invoke-RestMethod -uri "$RawSource" $VerCheck = $SourceCheck -split '\n' | Select-String -Pattern ".VERSION $ScriptVersion" -SimpleMatch -CaseSensitive -Quiet If ($VerCheck -ne $True) { Write-Log -Type Conf -Evt "*** There is an update available. ***" } } catch { } } ## Check for the apps list file, if it exists then sanitise it and if it doesn't exist then report and exit. If ($AppListFile) { If (Test-Path -Path $AppListFile) { $AppsList = Get-Content $AppListFile | Where-Object {$_.trim() -ne ""} } else { Write-Log -Type Err -Evt "The app list file $AppListFile does not exist." Exit } } If ($null -eq $AppListFile -And $PCApps -eq $false -And $UserApps -eq $false) { Write-Log -Type Err -Evt "No app list specified." Exit } ## Getting Windows Version info $OSVMaj = [environment]::OSVersion.Version | Select-Object -expand major $OSVMin = [environment]::OSVersion.Version | Select-Object -expand minor $OSVBui = [environment]::OSVersion.Version | Select-Object -expand build $OSV = "$OSVMaj" + "." + "$OSVMin" + "." + "$OSVBui" ## ## Display the current config and log if configured. ## Write-Log -Type Conf -Evt "--- Running with the following config ---" Write-Log -Type Conf -Evt "Utility Version: 24.11.09" UpdateCheck ## Run Update checker function Write-Log -Type Conf -Evt "Hostname: $Env:ComputerName." Write-Log -Type Conf -Evt "Windows Version: $OSV." If ($AppListFile) { Write-Log -Type Conf -Evt "Using list from file: $AppListFile." } If ($WimFile) { Write-Log -Type Conf -Evt "Wim File: $WimFile." } If ($WIndex) { Write-Log -Type Conf -Evt "Wim Index: $WIndex." } If ($WimMntPath) { Write-Log -Type Conf -Evt "Wim Mount Path: $WimMntPath." } If ($LogPathUsr) { Write-Log -Type Conf -Evt "Logs directory: $LogPath." } If ($Null -ne $LogHistory) { Write-Log -Type Conf -Evt "Logs to keep: $LogHistory days" } If ($PCOnly) { Write-Log -Type Conf -Evt "-PCOnly option is: $PCOnly." } If ($Uno) { Write-Log -Type Conf -Evt "-Uno option is: $Uno." } If ($AppListFile) { If ($Uno -eq $false) { Write-Log -Type Conf -Evt "Apps to remove:" ForEach ($App in $AppsList) { Write-Log -Type Conf -Evt "$App" } } else { Write-Log -Type Conf -Evt "Apps to keep:" ForEach ($App in $AppsList) { Write-Log -Type Conf -Evt "$App" } } } Write-Log -Type Conf -Evt "---" Write-Log -Type Info -Evt "Process started" ## ## Display current config ends here. ## ## ## Online Mode ## If ($Null -eq $WimFile) { If ($Uno -eq $false) { ## Remove the Apps listed in the file or report if app not present. ForEach ($App in $AppsList) { $PackageFullName = (Get-AppxPackage $App).PackageFullName $ProPackageFullName = (Get-AppxProvisionedPackage -Online | Where-Object {$_.DisplayName -eq $App}).PackageName If ($PCOnly -eq $false) { If ($PackageFullName) { Write-Log -Type Info -Evt "Removing Package: $App" Remove-AppxPackage -Package $PackageFullName | Out-Null } else { Write-Log -Type Info -Evt "Unable to find package: $App" } } If ($ProPackageFullName) { Write-Log -Type Info -Evt "Removing Provisioned Package: $ProPackageFullName" Remove-AppxProvisionedPackage -Online -PackageName $ProPackageFullName | Out-Null } else { Write-Log -Type Info -Evt "Unable to find provisioned package: $App" } } } else { $AppsToKeep = $AppsList -join "|" $AppsFullName = Get-AppxPackage | Where-Object {$_.PackageFullName -NotMatch $AppsToKeep} | Select-Object -ExpandProperty PackageFullName $ProvAppsFullName = Get-AppxProvisionedPackage -Online | Where-Object {$_.DisplayName -NotMatch $AppsToKeep} | Select-Object -ExpandProperty PackageName If ($PCOnly -eq $false) { ForEach ($AppFullName in $AppsFullName) { Write-Log -Type Info -Evt "Removing Package: $AppFullName" Remove-AppxPackage -Package $AppFullName | Out-Null } } ForEach ($ProvAppFullName in $ProvAppsFullName) { Write-Log -Type Info -Evt "Removing Provisioned Package: $ProvAppFullName" Remove-AppxProvisionedPackage -Online -PackageName $ProvAppFullName | Out-Null } } } ## TODO Uno mode for WIM Image ## ## Offline Mode ## If ($WimFile) { ## Default Wim Mount Path if none is configured. If ($Null -eq $WimMntPath) { $WimMntPath = "$Env:temp\RemMSStoreApps-WimMount" } ## Make sure the mount directory exists, if it doesn't create it. If ((Test-Path -Path $WimMntPath) -eq $False) { New-Item $WimMntPath -ItemType Directory -Force | Out-Null } ## Mount the Image. Mount-WindowsImage -ImagePath $WimFile -Index $WIndex -Path $WimMntPath | Out-Null If ($Uno -eq $false) { ## Remove the Apps listed above or report if app not present. ForEach ($App in $AppsList) { $ProPackageFullName = (Get-AppxProvisionedPackage -Path $WimMntPath | Where-Object {$_.DisplayName -eq $App}).PackageName If ($ProPackageFullName) { Write-Log -Type Info -Evt "Removing Provisioned Package: $ProPackageFullName" Remove-AppxProvisionedPackage -Path $WimMntPath -PackageName $ProPackageFullName | Out-Null } else { Write-Log -Type Info -Evt "Unable to find provisioned package: $App" } } } else { $AppsToKeep = $AppsList -join "|" $ProvAppsFullName = Get-AppxProvisionedPackage -Path $WimMntPath | Where-Object {$_.DisplayName -NotMatch $AppsToKeep} | Select-Object -ExpandProperty PackageName ForEach ($ProvAppFullName in $ProvAppsFullName) { Write-Log -Type Info -Evt "Removing Provisioned Package: $ProvAppFullName" Remove-AppxProvisionedPackage -Path $WimMntPath -PackageName $ProvAppFullName | Out-Null } } ## Dismount the image and save changes. Dismount-WindowsImage -Path $WimMntPath -Save | Out-Null } If ($PCApps) { Get-AppxProvisionedPackage -Online | Select-Object DisplayName | Format-Table -HideTableHeaders If ($LogPathUsr) { Get-AppxProvisionedPackage -Online | Select-Object DisplayName | Format-Table -HideTableHeaders | Out-File -Append $Log -Encoding ASCII } } If ($UserApps) { Get-AppxPackage | Select-Object Name | Format-Table -HideTableHeaders If ($LogPathUsr) { Get-AppxPackage | Select-Object Name | Format-Table -HideTableHeaders | Out-File -Append $Log -Encoding ASCII } } Write-Log -Type Info -Evt "Process finished." If ($Null -ne $LogHistory) { ## Cleanup logs. Write-Log -Type Info -Evt "Deleting logs older than: $LogHistory days" Get-ChildItem -Path "$LogPath\Remove-MS-Store-Apps_*" -File | Where-Object CreationTime -lt (Get-Date).AddDays(-$LogHistory) | Remove-Item -Recurse } } ## End |