param ( [Parameter(HelpMessage = "Change recommended version of Spotify.")] [Alias("v")] [string]$version, [Parameter(HelpMessage = "Use github.io mirror instead of raw.githubusercontent.")] [Alias("m")] [switch]$mirror, [Parameter(HelpMessage = "Developer mode activation.")] [Alias("dev")] [switch]$devtools, [Parameter(HelpMessage = 'Hiding podcasts/episodes/audiobooks from homepage.')] [switch]$podcasts_off, [Parameter(HelpMessage = 'Hiding Ad-like sections from the homepage')] [switch]$adsections_off, [Parameter(HelpMessage = 'Do not hiding podcasts/episodes/audiobooks from homepage.')] [switch]$podcasts_on, [Parameter(HelpMessage = 'Block Spotify automatic updates.')] [switch]$block_update_on, [Parameter(HelpMessage = 'Do not block Spotify automatic updates.')] [switch]$block_update_off, [Parameter(HelpMessage = 'Change limit for clearing audio cache.')] [Alias('cl')] [int]$cache_limit, [Parameter(HelpMessage = 'Automatic uninstallation of Spotify MS if it was found.')] [switch]$confirm_uninstall_ms_spoti, [Parameter(HelpMessage = 'Overwrite outdated or unsupported version of Spotify with the recommended version.')] [Alias('sp-over')] [switch]$confirm_spoti_recomended_over, [Parameter(HelpMessage = 'Uninstall outdated or unsupported version of Spotify and install the recommended version.')] [Alias('sp-uninstall')] [switch]$confirm_spoti_recomended_uninstall, [Parameter(HelpMessage = 'Installation without ad blocking for premium accounts.')] [switch]$premium, [Parameter(HelpMessage = 'Automatic launch of Spotify after installation is complete.')] [switch]$start_spoti, [Parameter(HelpMessage = 'Experimental features operated by Spotify.')] [switch]$exp_spotify, [Parameter(HelpMessage = 'Enable top search bar.')] [switch]$topsearchbar, [Parameter(HelpMessage = 'disable subfeed filter chips on home.')] [switch]$homesub_off, [Parameter(HelpMessage = 'Do not hide the icon of collaborations in playlists.')] [switch]$hide_col_icon_off, [Parameter(HelpMessage = 'Disable new right sidebar.')] [switch]$rightsidebar_off, [Parameter(HelpMessage = 'it`s killing the heart icon, you`re able to save and choose the destination for any song, playlist, or podcast')] [switch]$plus, [Parameter(HelpMessage = 'Enabled the big cards for home page')] [switch]$canvasHome, [Parameter(HelpMessage = 'Enable funny progress bar.')] [switch]$funnyprogressBar, [Parameter(HelpMessage = 'New theme activated (new right and left sidebar, some cover change)')] [switch]$new_theme, [Parameter(HelpMessage = 'Enable right sidebar coloring to match cover color)')] [switch]$rightsidebarcolor, [Parameter(HelpMessage = 'Returns old lyrics')] [switch]$old_lyrics, [Parameter(HelpMessage = 'Disable native lyrics')] [switch]$lyrics_block, [Parameter(HelpMessage = 'Do not create desktop shortcut.')] [switch]$no_shortcut, [Parameter(HelpMessage = 'Static color for lyrics.')] [ArgumentCompleter({ param($cmd, $param, $wordToComplete) [array] $validValues = @('default', 'red', 'orange', 'yellow', 'spotify', 'blue', 'purple', 'strawberry', 'pumpkin', 'sandbar', 'radium', 'oceano', 'royal', 'github', 'discord', 'drot', 'forest', 'fresh', 'zing', 'pinkle', 'krux', 'blueberry', 'postlight', 'relish', 'turquoise') $validValues -like "*$wordToComplete*" })] [string]$lyrics_stat, [Parameter(HelpMessage = 'Accumulation of track listening history with Goofy.')] [string]$urlform_goofy = $null, [Parameter(HelpMessage = 'Accumulation of track listening history with Goofy.')] [string]$idbox_goofy = $null, [Parameter(HelpMessage = 'Error log ru string.')] [switch]$err_ru, [Parameter(HelpMessage = 'Select the desired language to use for installation. Default is the detected system language.')] [Alias('l')] [string]$language ) # Ignore errors from `Stop-Process` $PSDefaultParameterValues['Stop-Process:ErrorAction'] = [System.Management.Automation.ActionPreference]::SilentlyContinue function Format-LanguageCode { # Normalizes and confirms support of the selected language. [CmdletBinding()] [OutputType([string])] param ( [string]$LanguageCode ) $supportLanguages = @( 'en', 'ru', 'it', 'tr', 'ka', 'pl', 'es', 'fr', 'hi', 'pt', 'id', 'vi', 'ro', 'de', 'hu', 'zh', 'zh-TW', 'ko', 'ua', 'fa', 'sr', 'lv', 'bn', 'el', 'fi', 'ja', 'fil', 'sv', 'sk', 'ta', 'cs' ) # Trim the language code down to two letter code. switch -Regex ($LanguageCode) { '^en' { $returnCode = 'en' break } '^(ru|py)' { $returnCode = 'ru' break } '^it' { $returnCode = 'it' break } '^tr' { $returnCode = 'tr' break } '^ka' { $returnCode = 'ka' break } '^pl' { $returnCode = 'pl' break } '^es' { $returnCode = 'es' break } '^fr' { $returnCode = 'fr' break } '^hi' { $returnCode = 'hi' break } '^pt' { $returnCode = 'pt' break } '^id' { $returnCode = 'id' break } '^vi' { $returnCode = 'vi' break } '^ro' { $returnCode = 'ro' break } '^de' { $returnCode = 'de' break } '^hu' { $returnCode = 'hu' break } '^(zh|zh-CN)$' { $returnCode = 'zh' break } '^zh-TW' { $returnCode = 'zh-TW' break } '^ko' { $returnCode = 'ko' break } '^ua' { $returnCode = 'ua' break } '^fa' { $returnCode = 'fa' break } '^sr' { $returnCode = 'sr' break } '^lv' { $returnCode = 'lv' break } '^bn' { $returnCode = 'bn' break } '^el' { $returnCode = 'el' break } '^fi$' { $returnCode = 'fi' break } '^ja' { $returnCode = 'ja' break } '^fil' { $returnCode = 'fil' break } '^sv' { $returnCode = 'sv' break } '^sk' { $returnCode = 'sk' break } '^ta' { $returnCode = 'ta' break } '^cs' { $returnCode = 'cs' break } Default { $returnCode = $PSUICulture $long_code = $true break } } # Checking the long language code if ($long_code -and $returnCode -NotIn $supportLanguages) { $returnCode = $returnCode -split "-" | Select-Object -First 1 } # Checking the short language code if ($returnCode -NotIn $supportLanguages) { # If the language code is not supported default to English. $returnCode = 'en' } return $returnCode } $spotifyDirectory = Join-Path $env:APPDATA 'Spotify' $spotifyDirectory2 = Join-Path $env:LOCALAPPDATA 'Spotify' $spotifyExecutable = Join-Path $spotifyDirectory 'Spotify.exe' $exe_bak = Join-Path $spotifyDirectory 'Spotify.bak' $spotifyUninstall = Join-Path ([System.IO.Path]::GetTempPath()) 'SpotifyUninstall.exe' $start_menu = Join-Path $env:APPDATA 'Microsoft\Windows\Start Menu\Programs\Spotify.lnk' $upgrade_client = $false # Check version Powershell $psv = $PSVersionTable.PSVersion.major if ($psv -ge 7) { Import-Module Appx -UseWindowsPowerShell -WarningAction:SilentlyContinue } # add Tls12 [Net.ServicePointManager]::SecurityProtocol = 3072 function Get-Link { param ( [Alias("e")] [string]$endlink ) switch ($mirror) { $true { return "https://spotx-official.github.io/SpotX" + $endlink } default { return "https://raw.githubusercontent.com/SpotX-Official/SpotX/main" + $endlink } } } function CallLang($clg) { $ProgressPreference = 'SilentlyContinue' try { $response = (iwr -Uri (Get-Link -e "/scripts/installer-lang/$clg.ps1") -UseBasicParsing).Content if ($mirror) { $response = [System.Text.Encoding]::UTF8.GetString($response) } Invoke-Expression $response } catch { Write-Host "Error loading $clg language" Pause Exit } } # Set language code for script. $langCode = Format-LanguageCode -LanguageCode $Language $lang = CallLang -clg $langCode Write-Host ($lang).Welcome Write-Host # Check version Windows $os = Get-CimInstance -ClassName "Win32_OperatingSystem" -ErrorAction SilentlyContinue if ($os) { $osCaption = $os.Caption } else { $osCaption = (Get-ItemProperty -Path "HKLM:\SOFTWARE\Microsoft\Windows NT\CurrentVersion" -Name ProductName).ProductName } $pattern = "\bWindows (7|8(\.1)?|10|11|12)\b" $reg = [regex]::Matches($osCaption, $pattern) $win_os = $reg.Value $win12 = $win_os -match "\windows 12\b" $win11 = $win_os -match "\windows 11\b" $win10 = $win_os -match "\windows 10\b" $win8_1 = $win_os -match "\windows 8.1\b" $win8 = $win_os -match "\windows 8\b" $win7 = $win_os -match "\windows 7\b" $match_v = "^\d+\.\d+\.\d+\.\d+\.g[0-9a-f]{8}-\d+$" if ($version) { if ($version -match $match_v) { $onlineFull = $version } else { Write-Warning "Invalid $($version) format. Example: 1.2.13.661.ga588f749-4064" Write-Host } } $old_os = $win7 -or $win8 -or $win8_1 # Recommended version for Win 7-8.1 $last_win7_full = "1.2.5.1006.g22820f93-1078" if (!($version -and $version -match $match_v)) { if ($old_os) { $onlineFull = $last_win7_full } else { # Recommended version for Win 10-12 $onlineFull = "1.2.46.462.gf57913e0-290" } } else { if ($old_os) { $last_win7 = "1.2.5.1006" if ([version]($onlineFull -split ".g")[0] -gt [version]$last_win7) { Write-Warning ("Version {0} is only supported on Windows 10 and above" -f ($onlineFull -split ".g")[0]) Write-Warning ("The recommended version has been automatically changed to {0}, the latest supported version for Windows 7-8.1" -f $last_win7) Write-Host $onlineFull = $last_win7_full } } } $online = ($onlineFull -split ".g")[0] function Get { param ( [string]$Url, [int]$MaxRetries = 3, [int]$RetrySeconds = 3 ) $retries = 0 while ($retries -lt $MaxRetries) { try { return Invoke-RestMethod -Uri $Url } catch { Write-Warning "Request failed: $_" $retries++ Start-Sleep -Seconds $RetrySeconds } } Write-Host Write-Host "ERROR: " -ForegroundColor Red -NoNewline; Write-Host "Failed to retrieve data from $Url" -ForegroundColor White Write-Host return $null } function incorrectValue { Write-Host ($lang).Incorrect"" -ForegroundColor Red -NoNewline Write-Host ($lang).Incorrect2"" -NoNewline Start-Sleep -Milliseconds 1000 Write-Host "3" -NoNewline Start-Sleep -Milliseconds 1000 Write-Host " 2" -NoNewline Start-Sleep -Milliseconds 1000 Write-Host " 1" Start-Sleep -Milliseconds 1000 Clear-Host } function Unlock-Folder { $blockFileUpdate = Join-Path $env:LOCALAPPDATA 'Spotify\Update' if (Test-Path $blockFileUpdate -PathType Container) { $folderUpdateAccess = Get-Acl $blockFileUpdate $hasDenyAccessRule = $false foreach ($accessRule in $folderUpdateAccess.Access) { if ($accessRule.AccessControlType -eq 'Deny') { $hasDenyAccessRule = $true $folderUpdateAccess.RemoveAccessRule($accessRule) } } if ($hasDenyAccessRule) { Set-Acl $blockFileUpdate $folderUpdateAccess } } } function Mod-F { param( [string] $template, [object[]] $arguments ) $result = $template for ($i = 0; $i -lt $arguments.Length; $i++) { $placeholder = "{${i}}" $value = $arguments[$i] $result = $result -replace [regex]::Escape($placeholder), $value } return $result } function downloadSp() { $webClient = New-Object -TypeName System.Net.WebClient Import-Module BitsTransfer $web_Url = "https://download.scdn.co/upgrade/client/win32-x86/spotify_installer-$onlineFull.exe" $local_Url = "$PWD\SpotifySetup.exe" $web_name_file = "SpotifySetup.exe" try { if (curl.exe -V) { $curl_check = $true } } catch { $curl_check = $false } try { if ($curl_check) { $stcode = curl.exe -Is -w "%{http_code} \n" -o /dev/null $web_Url --retry 2 --ssl-no-revoke if ($stcode.trim() -ne "200") { Write-Host "Curl error code: $stcode"; throw } curl.exe -q $web_Url -o $local_Url --progress-bar --retry 3 --ssl-no-revoke return } if (!($curl_check ) -and $null -ne (Get-Module -Name BitsTransfer -ListAvailable)) { $ProgressPreference = 'Continue' Start-BitsTransfer -Source $web_Url -Destination $local_Url -DisplayName ($lang).Download5 -Description "$online " return } if (!($curl_check ) -and $null -eq (Get-Module -Name BitsTransfer -ListAvailable)) { $webClient.DownloadFile($web_Url, $local_Url) return } } catch { Write-Host Write-Host ($lang).Download $web_name_file -ForegroundColor RED $Error[0].Exception Write-Host Write-Host ($lang).Download2`n Start-Sleep -Milliseconds 5000 try { if ($curl_check) { $stcode = curl.exe -Is -w "%{http_code} \n" -o /dev/null $web_Url --retry 2 --ssl-no-revoke if ($stcode.trim() -ne "200") { Write-Host "Curl error code: $stcode"; throw } curl.exe -q $web_Url -o $local_Url --progress-bar --retry 3 --ssl-no-revoke return } if (!($curl_check ) -and $null -ne (Get-Module -Name BitsTransfer -ListAvailable) -and !($curl_check )) { Start-BitsTransfer -Source $web_Url -Destination $local_Url -DisplayName ($lang).Download5 -Description "$online " return } if (!($curl_check ) -and $null -eq (Get-Module -Name BitsTransfer -ListAvailable) -and !($curl_check )) { $webClient.DownloadFile($web_Url, $local_Url) return } } catch { Write-Host ($lang).Download3 -ForegroundColor RED $Error[0].Exception Write-Host Write-Host ($lang).Download4`n ($lang).StopScript $tempDirectory = $PWD Pop-Location Start-Sleep -Milliseconds 200 Remove-Item -Recurse -LiteralPath $tempDirectory Pause Exit } } } function DesktopFolder { # If the default Dekstop folder does not exist, then try to find it through the registry. $ErrorActionPreference = 'SilentlyContinue' if (Test-Path "$env:USERPROFILE\Desktop") { $desktop_folder = "$env:USERPROFILE\Desktop" } $regedit_desktop_folder = Get-ItemProperty -Path "Registry::HKEY_CURRENT_USER\SOFTWARE\Microsoft\Windows\CurrentVersion\Explorer\User Shell Folders\" $regedit_desktop = $regedit_desktop_folder.'{754AC886-DF64-4CBA-86B5-F7FBF4FBCEF5}' if (!(Test-Path "$env:USERPROFILE\Desktop")) { $desktop_folder = $regedit_desktop } return $desktop_folder } function Kill-Spotify { param ( [int]$maxAttempts = 5 ) for ($attempt = 1; $attempt -le $maxAttempts; $attempt++) { $allProcesses = Get-Process -ErrorAction SilentlyContinue $spotifyProcesses = $allProcesses | Where-Object { $_.ProcessName -like "*spotify*" } if ($spotifyProcesses) { foreach ($process in $spotifyProcesses) { try { Stop-Process -Id $process.Id -Force } catch { # Ignore NoSuchProcess exception } } Start-Sleep -Seconds 1 } else { break } } if ($attempt -gt $maxAttempts) { Write-Host "The maximum number of attempts to terminate a process has been reached." } } Kill-Spotify # Remove Spotify Windows Store If Any if ($win10 -or $win11 -or $win8_1 -or $win8 -or $win12) { if (Get-AppxPackage -Name SpotifyAB.SpotifyMusic) { Write-Host ($lang).MsSpoti`n if (!($confirm_uninstall_ms_spoti)) { do { $ch = Read-Host -Prompt ($lang).MsSpoti2 Write-Host if (!($ch -eq 'n' -or $ch -eq 'y')) { incorrectValue } } while ($ch -notmatch '^y$|^n$') } if ($confirm_uninstall_ms_spoti) { $ch = 'y' } if ($ch -eq 'y') { $ProgressPreference = 'SilentlyContinue' # Hiding Progress Bars if ($confirm_uninstall_ms_spoti) { Write-Host ($lang).MsSpoti3`n } if (!($confirm_uninstall_ms_spoti)) { Write-Host ($lang).MsSpoti4`n } Get-AppxPackage -Name SpotifyAB.SpotifyMusic | Remove-AppxPackage } if ($ch -eq 'n') { Read-Host ($lang).StopScript Pause Exit } } } # Attempt to fix the hosts file $hostsFilePath = Join-Path $Env:windir 'System32\Drivers\Etc\hosts' $hostsBackupFilePath = Join-Path $Env:windir 'System32\Drivers\Etc\hosts.bak' if (Test-Path -Path $hostsFilePath) { $hosts = [System.IO.File]::ReadAllLines($hostsFilePath) $regex = "^(?!#|\|)((?:.*?(?:download|upgrade)\.scdn\.co|.*?spotify).*)" if ($hosts -match $regex) { Write-Host ($lang).HostInfo`n Write-Host ($lang).HostBak`n Copy-Item -Path $hostsFilePath -Destination $hostsBackupFilePath -ErrorAction SilentlyContinue if ($?) { Write-Host ($lang).HostDel try { $hosts = $hosts | Where-Object { $_ -notmatch $regex } [System.IO.File]::WriteAllLines($hostsFilePath, $hosts) } catch { Write-Host ($lang).HostError`n -ForegroundColor Red $copyError = $Error[0] Write-Host "Error: $($copyError.Exception.Message)`n" -ForegroundColor Red } } else { Write-Host ($lang).HostError`n -ForegroundColor Red $copyError = $Error[0] Write-Host "Error: $($copyError.Exception.Message)`n" -ForegroundColor Red } } } # Unique directory name based on time Push-Location -LiteralPath ([System.IO.Path]::GetTempPath()) New-Item -Type Directory -Name "SpotX_Temp-$(Get-Date -UFormat '%Y-%m-%d_%H-%M-%S')" | Convert-Path | Set-Location if ($premium) { Write-Host ($lang).Prem`n } $spotifyInstalled = (Test-Path -LiteralPath $spotifyExecutable) if ($spotifyInstalled) { # Check version Spotify offline $offline = (Get-Item $spotifyExecutable).VersionInfo.FileVersion # Version comparison # converting strings to arrays of numbers using the -split operator and a foreach loop $arr1 = $online -split '\.' | foreach { [int]$_ } $arr2 = $offline -split '\.' | foreach { [int]$_ } # compare each element of the array in order from most significant to least significant. for ($i = 0; $i -lt $arr1.Length; $i++) { if ($arr1[$i] -gt $arr2[$i]) { $oldversion = $true break } elseif ($arr1[$i] -lt $arr2[$i]) { $testversion = $true break } } # Old version Spotify if ($oldversion) { if ($confirm_spoti_recomended_over -or $confirm_spoti_recomended_uninstall) { Write-Host ($lang).OldV`n } if (!($confirm_spoti_recomended_over) -and !($confirm_spoti_recomended_uninstall)) { do { Write-Host (($lang).OldV2 -f $offline, $online) $ch = Read-Host -Prompt ($lang).OldV3 Write-Host if (!($ch -eq 'n' -or $ch -eq 'y')) { incorrectValue } } while ($ch -notmatch '^y$|^n$') } if ($confirm_spoti_recomended_over -or $confirm_spoti_recomended_uninstall) { $ch = 'y' Write-Host ($lang).AutoUpd`n } if ($ch -eq 'y') { $upgrade_client = $true if (!($confirm_spoti_recomended_over) -and !($confirm_spoti_recomended_uninstall)) { do { $ch = Read-Host -Prompt (($lang).DelOrOver -f $offline) Write-Host if (!($ch -eq 'n' -or $ch -eq 'y')) { incorrectValue } } while ($ch -notmatch '^y$|^n$') } if ($confirm_spoti_recomended_uninstall) { $ch = 'y' } if ($confirm_spoti_recomended_over) { $ch = 'n' } if ($ch -eq 'y') { Write-Host ($lang).DelOld`n $null = Unlock-Folder cmd /c $spotifyExecutable /UNINSTALL /SILENT wait-process -name SpotifyUninstall Start-Sleep -Milliseconds 200 if (Test-Path $spotifyDirectory) { Remove-Item -Recurse -Force -LiteralPath $spotifyDirectory } if (Test-Path $spotifyDirectory2) { Remove-Item -Recurse -Force -LiteralPath $spotifyDirectory2 } if (Test-Path $spotifyUninstall ) { Remove-Item -Recurse -Force -LiteralPath $spotifyUninstall } } if ($ch -eq 'n') { $ch = $null } } if ($ch -eq 'n') { $downgrading = $true } } # Unsupported version Spotify if ($testversion) { # Submit unsupported version of Spotify to google form for further processing try { # Country check $country = [System.Globalization.RegionInfo]::CurrentRegion.EnglishName $txt = [IO.File]::ReadAllText($spotifyExecutable) $regex = "(\d+)\.(\d+)\.(\d+)\.(\d+)(\.g[0-9a-f]{8})" $v = $txt | Select-String $regex -AllMatches $ver = $v.Matches.Value[0] if ($ver.Count -gt 1) { $ver = $ver[0] } $Parameters = @{ Uri = 'https://docs.google.com/forms/d/e/1FAIpQLSegGsAgilgQ8Y36uw-N7zFF6Lh40cXNfyl1ecHPpZcpD8kdHg/formResponse' Method = 'POST' Body = @{ 'entry.620327948' = $ver 'entry.1951747592' = $country 'entry.1402903593' = $win_os 'entry.860691305' = $psv 'entry.2067427976' = $online + " < " + $offline } } $null = Invoke-WebRequest -useb @Parameters } catch { Write-Host 'Unable to submit new version of Spotify' Write-Host "error description: "$Error[0] } if ($confirm_spoti_recomended_over -or $confirm_spoti_recomended_uninstall) { Write-Host ($lang).NewV`n } if (!($confirm_spoti_recomended_over) -and !($confirm_spoti_recomended_uninstall)) { do { Write-Host (($lang).NewV2 -f $offline, $online) $ch = Read-Host -Prompt (($lang).NewV3 -f $offline) Write-Host if (!($ch -eq 'n' -or $ch -eq 'y')) { incorrectValue } } while ($ch -notmatch '^y$|^n$') } if ($confirm_spoti_recomended_over -or $confirm_spoti_recomended_uninstall) { $ch = 'n' } if ($ch -eq 'y') { $upgrade_client = $false } if ($ch -eq 'n') { if (!($confirm_spoti_recomended_over) -and !($confirm_spoti_recomended_uninstall)) { do { $ch = Read-Host -Prompt (($lang).Recom -f $online) Write-Host if (!($ch -eq 'n' -or $ch -eq 'y')) { incorrectValue } } while ($ch -notmatch '^y$|^n$') } if ($confirm_spoti_recomended_over -or $confirm_spoti_recomended_uninstall) { $ch = 'y' Write-Host ($lang).AutoUpd`n } if ($ch -eq 'y') { $upgrade_client = $true $downgrading = $true if (!($confirm_spoti_recomended_over) -and !($confirm_spoti_recomended_uninstall)) { do { $ch = Read-Host -Prompt (($lang).DelOrOver -f $offline) Write-Host if (!($ch -eq 'n' -or $ch -eq 'y')) { incorrectValue } } while ($ch -notmatch '^y$|^n$') } if ($confirm_spoti_recomended_uninstall) { $ch = 'y' } if ($confirm_spoti_recomended_over) { $ch = 'n' } if ($ch -eq 'y') { Write-Host ($lang).DelNew`n $null = Unlock-Folder cmd /c $spotifyExecutable /UNINSTALL /SILENT wait-process -name SpotifyUninstall Start-Sleep -Milliseconds 200 if (Test-Path $spotifyDirectory) { Remove-Item -Recurse -Force -LiteralPath $spotifyDirectory } if (Test-Path $spotifyDirectory2) { Remove-Item -Recurse -Force -LiteralPath $spotifyDirectory2 } if (Test-Path $spotifyUninstall ) { Remove-Item -Recurse -Force -LiteralPath $spotifyUninstall } } if ($ch -eq 'n') { $ch = $null } } if ($ch -eq 'n') { Write-Host ($lang).StopScript $tempDirectory = $PWD Pop-Location Start-Sleep -Milliseconds 200 Remove-Item -Recurse -LiteralPath $tempDirectory Pause Exit } } } } # If there is no client or it is outdated, then install if (-not $spotifyInstalled -or $upgrade_client) { Write-Host ($lang).DownSpoti"" -NoNewline Write-Host $online -ForegroundColor Green Write-Host ($lang).DownSpoti2`n # Delete old version files of Spotify before installing, leave only profile files $ErrorActionPreference = 'SilentlyContinue' Kill-Spotify Start-Sleep -Milliseconds 600 $null = Unlock-Folder Start-Sleep -Milliseconds 200 Get-ChildItem $spotifyDirectory -Exclude 'Users', 'prefs' | Remove-Item -Recurse -Force Start-Sleep -Milliseconds 200 # Client download downloadSp Write-Host Start-Sleep -Milliseconds 200 # Client installation Start-Process -FilePath explorer.exe -ArgumentList $PWD\SpotifySetup.exe while (-not (get-process | Where-Object { $_.ProcessName -eq 'SpotifySetup' })) {} wait-process -name SpotifySetup Kill-Spotify # Upgrade check version Spotify offline $offline = (Get-Item $spotifyExecutable).VersionInfo.FileVersion # Upgrade check version Spotify.bak $offline_bak = (Get-Item $exe_bak).VersionInfo.FileVersion } # Delete Spotify shortcut if it is on desktop if ($no_shortcut) { $ErrorActionPreference = 'SilentlyContinue' $desktop_folder = DesktopFolder Start-Sleep -Milliseconds 1000 remove-item "$desktop_folder\Spotify.lnk" -Recurse -Force } $ch = $null # updated Russian translation if ($langCode -eq 'ru' -and [version]$offline -ge [version]"1.1.92.644") { $webjsonru = Get -Url (Get-Link -e "/patches/Augmented%20translation/ru.json") if ($webjsonru -ne $null) { $ru = $true } } if ($podcasts_off) { Write-Host ($lang).PodcatsOff`n $ch = 'y' } if ($podcasts_on) { Write-Host ($lang).PodcastsOn`n $ch = 'n' } if (!($podcasts_off) -and !($podcasts_on)) { do { $ch = Read-Host -Prompt ($lang).PodcatsSelect Write-Host if (!($ch -eq 'n' -or $ch -eq 'y')) { incorrectValue } } while ($ch -notmatch '^y$|^n$') } if ($ch -eq 'y') { $podcast_off = $true } $ch = $null if ($downgrading) { $upd = "`n" + [string]($lang).DowngradeNote } else { $upd = "" } if ($block_update_on) { Write-Host ($lang).UpdBlock`n $ch = 'y' } if ($block_update_off) { Write-Host ($lang).UpdUnblock`n $ch = 'n' } if (!($block_update_on) -and !($block_update_off)) { do { $text_upd = [string]($lang).UpdSelect + $upd $ch = Read-Host -Prompt $text_upd Write-Host if (!($ch -eq 'n' -or $ch -eq 'y')) { incorrectValue } } while ($ch -notmatch '^y$|^n$') } if ($ch -eq 'y') { $not_block_update = $false } if (!($new_theme) -and [version]$offline -ge [version]"1.2.14.1141") { Write-Warning "This version does not support the old theme, use version 1.2.13.661 or below" Write-Host } if ($ch -eq 'n') { $not_block_update = $true $ErrorActionPreference = 'SilentlyContinue' if ((Test-Path -LiteralPath $exe_bak) -and $offline -eq $offline_bak) { Remove-Item $spotifyExecutable -Recurse -Force Rename-Item $exe_bak $spotifyExecutable } } $ch = $null $webjson = Get -Url (Get-Link -e "/patches/patches.json") -RetrySeconds 5 if ($webjson -eq $null) { Write-Host Write-Host "Failed to get patches.json" -ForegroundColor Red Write-Host ($lang).StopScript $tempDirectory = $PWD Pop-Location Start-Sleep -Milliseconds 200 Remove-Item -Recurse -LiteralPath $tempDirectory Pause Exit } function Helper($paramname) { function Remove-Json { param ( [Parameter(Mandatory = $true)] [Alias("j")] [PSObject]$Json, [Parameter(Mandatory = $true, ValueFromPipeline = $true)] [Alias("p")] [string[]]$Properties ) foreach ($Property in $Properties) { $Json.psobject.properties.Remove($Property) } } function Move-Json { param ( [Parameter(Mandatory = $true)] [Alias("t")] [PSObject]$to, [Parameter(Mandatory = $true)] [Alias("n")] [string[]]$name, [Parameter(Mandatory = $true)] [Alias("f")] [PSObject]$from ) foreach ($propertyName in $name) { $from | Add-Member -MemberType NoteProperty -Name $propertyName -Value $to.$propertyName Remove-Json -j $to -p $propertyName } } switch ( $paramname ) { "HtmlLicMin" { # licenses.html minification $name = "patches.json.others." $n = "licenses.html" $contents = "htmlmin" $json = $webjson.others } "HtmlBlank" { # htmlBlank minification $name = "patches.json.others." $n = "blank.html" $contents = "blank.html" $json = $webjson.others } "MinJs" { # Minification of all *.js $contents = "minjs" $json = $webjson.others } "MinJson" { # Minification of all *.json $contents = "minjson" $json = $webjson.others } "FixCss" { # Remove indent for old theme xpui.css $name = "patches.json.others." $n = "xpui.css" $json = $webjson.others } "RemovertlCssmin" { # Remove RTL and minification of all *.css $contents = "removertl-cssmin" $json = $webjson.others } "DisableSentry" { # Disable Sentry (vendor~xpui.js) $name = "patches.json.others." $n = "vendor~xpui.js" $contents = "disablesentry" $json = $webjson.others } "Lyrics-color" { $pasttext = $webjson.others.themelyrics.theme.$lyrics_stat.pasttext $current = $webjson.others.themelyrics.theme.$lyrics_stat.current $next = $webjson.others.themelyrics.theme.$lyrics_stat.next $background = $webjson.others.themelyrics.theme.$lyrics_stat.background $hover = $webjson.others.themelyrics.theme.$lyrics_stat.hover $maxmatch = $webjson.others.themelyrics.theme.$lyrics_stat.maxmatch if ([version]$offline -lt [version]"1.1.99.871") { $lyrics = "lyricscolor1"; $contents = $lyrics } if ([version]$offline -ge [version]"1.1.99.871") { $lyrics = "lyricscolor2"; $contents = $lyrics } # xpui.js or xpui-routes-lyrics.js if ([version]$offline -ge [version]"1.1.99.871") { $full_previous = Mod-F -template $webjson.others.$lyrics.add[0] -arguments $pasttext $full_current = Mod-F -template $webjson.others.$lyrics.add[1] -arguments $current $full_next = Mod-F -template $webjson.others.$lyrics.add[2] -arguments $next $full_lyrics = Mod-F -template $webjson.others.$lyrics.add[3] -arguments $full_previous, $full_current, $full_next $webjson.others.$lyrics.add[3] = $full_lyrics $webjson.others.$lyrics.replace[1] = '$1' + '"' + $pasttext + '"' $webjson.others.$lyrics.replace[2] = '$1' + '"' + $current + '"' $webjson.others.$lyrics.replace[3] = '$1' + '"' + $next + '"' $webjson.others.$lyrics.replace[4] = '$1' + '"' + $background + '"' $webjson.others.$lyrics.replace[5] = '$1' + '"' + $hover + '"' $webjson.others.$lyrics.replace[6] = '$1' + '"' + $maxmatch + '"' if ([version]$offline -ge [version]"1.2.6.861") { $webjson.others.$lyrics.replace[7] = '$1' + '"' + $maxmatch + '"' + '$3' } else { $webjson.others.$lyrics.match = $webjson.others.$lyrics.match | Where-Object { $_ -ne $webjson.others.$lyrics.match[7] } } if ([version]$offline -ge [version]"1.2.3.1107") { $webjson.others.$lyrics.replace[8] = $webjson.others.$lyrics.replace[8] -f $background } } # xpui-routes-lyrics.css if ([version]$offline -lt [version]"1.1.99.871") { $webjson.others.$lyrics.replace[0] = '$1' + $pasttext $webjson.others.$lyrics.replace[1] = '$1' + $current $webjson.others.$lyrics.replace[2] = '$1' + $next $webjson.others.$lyrics.replace[3] = $background $webjson.others.$lyrics.replace[4] = '$1' + $hover $webjson.others.$lyrics.replace[5] = '$1' + $maxmatch } $name = "patches.json.others." $n = $name_file $json = $webjson.others } "Discriptions" { # Add discriptions (xpui-desktop-modals.js) $svg_tg = $webjson.others.discriptions.svgtg $svg_git = $webjson.others.discriptions.svggit $svg_faq = $webjson.others.discriptions.svgfaq $replace = $webjson.others.discriptions.replace $replacedText = $replace -f $svg_git, $svg_tg, $svg_faq $webjson.others.discriptions.replace = '$1"' + $replacedText + '"})' $name = "patches.json.others." $n = "xpui-desktop-modals.js" $contents = "discriptions" $json = $webjson.others } "OffadsonFullscreen" { # Full screen mode activation and removing "Upgrade to premium" menu, upgrade button, disabling a playlist sponsor $name = "patches.json.free." $n = "xpui.js" $contents = $webjson.free.psobject.properties.name $json = $webjson.free } "ForcedExp" { # Forced disable some exp (xpui.js) $offline_patch = $offline -replace '(\d+\.\d+\.\d+)(.\d+)', '$1' $Enable = $webjson.others.EnableExp $Disable = $webjson.others.DisableExp $Custom = $webjson.others.CustomExp # carousel is temporarily disabled because it causes lags in the main menu Move-Json -n 'HomeCarousels' -t $Enable -f $Disable if ($podcast_off) { Move-Json -n 'HomePin' -t $Enable -f $Disable } # disabled broken panel from 1.2.37 to 1.2.38 if ([version]$offline -eq [version]'1.2.37.701' -or [version]$offline -eq [version]'1.2.38.720' ) { Move-Json -n 'DevicePickerSidePanel' -t $Enable -f $Disable } if ([version]$offline -ge [version]'1.2.41.434' -and $lyrics_block) { Move-Json -n 'Lyrics' -t $Enable -f $Disable } if ([version]$offline -eq [version]'1.2.30.1135') { Move-Json -n 'QueueOnRightPanel' -t $Enable -f $Disable } if (!($plus)) { Move-Json -n "Plus", "AlignedCurationSavedIn" -t $Enable -f $Disable } if (!$topsearchbar -or [version]$offline -ge [version]"1.2.46.462") { Move-Json -n "GlobalNavBar" -t $Enable -f $Disable $Custom.GlobalNavBar.value = "control" } if (!($funnyprogressbar)) { Move-Json -n 'HeBringsNpb' -t $Enable -f $Disable } if (!($canvasHome)) { Move-Json -n "canvasHome", "canvasHomeAudioPreviews" -t $Enable -f $Disable } # disable subfeed filter chips on home if ($homesub_off) { Move-Json -n "HomeSubfeeds" -t $Enable -f $Disable } # Old theme if (!($new_theme) -and [version]$offline -le [version]"1.2.13.661") { Move-Json -n 'RightSidebar', 'LeftSidebar' -t $Enable -f $Disable Remove-Json -j $Custom -p "NavAlt", 'NavAlt2' Remove-Json -j $Enable -p 'RightSidebarLyrics', 'RightSidebarCredits', 'RightSidebar', 'LeftSidebar', 'RightSidebarColors' } # New theme else { if ($rightsidebar_off -and [version]$offline -lt [version]"1.2.24.756") { Move-Json -n 'RightSidebar' -t $Enable -from $Disable } else { if (!($rightsidebarcolor)) { Remove-Json -j $Enable -p 'RightSidebarColors' } if ($old_lyrics) { Remove-Json -j $Enable -p 'RightSidebarLyrics' } } } if (!$premium) { Remove-Json -j $Enable -p 'RemoteDownloads' } # Disable unimportant exp if ($exp_spotify) { $objects = @( @{ Object = $webjson.others.CustomExp.psobject.properties PropertiesToKeep = @('LyricsUpsell') }, @{ Object = $webjson.others.EnableExp.psobject.properties PropertiesToKeep = @('BrowseViaPathfinder', 'HomeViaGraphQLV2') } ) foreach ($obj in $objects) { $propertiesToRemove = $obj.Object.Name | Where-Object { $_ -notin $obj.PropertiesToKeep } $propertiesToRemove | foreach { $obj.Object.Remove($_) } } } $Exp = ($Enable, $Disable, $Custom) foreach ($item in $Exp) { $itemProperties = $item | Get-Member -MemberType NoteProperty | Select-Object -ExpandProperty Name foreach ($key in $itemProperties) { $vers = $item.$key.version if (!($vers.to -eq "" -or [version]$vers.to -ge [version]$offline_patch -and [version]$vers.fr -le [version]$offline_patch)) { if ($item.PSObject.Properties.Name -contains $key) { $item.PSObject.Properties.Remove($key) } } } } $Enable = $webjson.others.EnableExp $Disable = $webjson.others.DisableExp $Custom = $webjson.others.CustomExp $enableNames = foreach ($item in $Enable.PSObject.Properties.Name) { $webjson.others.EnableExp.$item.name } $disableNames = foreach ($item in $Disable.PSObject.Properties.Name) { $webjson.others.DisableExp.$item.name } $customNames = foreach ($item in $Custom.PSObject.Properties.Name) { $custname = $webjson.others.CustomExp.$item.name $custvalue = $webjson.others.CustomExp.$item.value # Create a string with the desired format $objectString = "{name:'$custname',value:'$custvalue'}" $objectString } # Convert the strings of objects into a single text string if ([string]::IsNullOrEmpty($customNames)) { $customTextVariable = '[]' } else { $customTextVariable = "[" + ($customNames -join ',') + "]" } if ([string]::IsNullOrEmpty($enableNames)) { $enableTextVariable = '[]' } else { $enableTextVariable = "['" + ($enableNames -join "','") + "']" } if ([string]::IsNullOrEmpty($disableNames)) { $disableTextVariable = '[]' } else { $disableTextVariable = "['" + ($disableNames -join "','") + "']" } $replacements = @( @("EnableExp=[]", "EnableExp=$enableTextVariable"), @("DisableExp=[]", "DisableExp=$disableTextVariable"), @("CustomExp=[]", "CustomExp=$customTextVariable") ) foreach ($replacement in $replacements) { $webjson.others.ForcedExp.replace = $webjson.others.ForcedExp.replace.Replace($replacement[0], $replacement[1]) } $name = "patches.json.others." $n = "xpui.js" $contents = "ForcedExp" $json = $webjson.others } "RuTranslate" { # Additional translation of some words for the Russian language $n = "ru.json" $contents = $webjsonru.psobject.properties.name $json = $webjsonru } "Binary" { $binary = $webjson.others.binary if ($not_block_update) { Remove-Json -j $binary -p 'block_update' } $name = "patches.json.others.binary." $n = "Spotify.exe" $contents = $webjson.others.binary.psobject.properties.name $json = $webjson.others.binary } "Collaborators" { # Hide Collaborators icon $name = "patches.json.others." $n = "xpui-routes-playlist.js" $contents = "collaboration" $json = $webjson.others } "Dev" { $name = "patches.json.others." $n = "xpui-routes-desktop-settings.js" $contents = "dev-tools" $json = $webjson.others } "VariousofXpui-js" { $VarJs = $webjson.VariousJs if ($topsearchbar -or ([version]$offline -ne [version]"1.2.45.451" -and [version]$offline -ne [version]"1.2.45.454")) { Remove-Json -j $VarJs -p "fixTitlebarHeight" } if (!($lyrics_block)) { Remove-Json -j $VarJs -p "lyrics-block" } else { Remove-Json -j $VarJs -p "lyrics-old-on" } if (!($devtools)) { Remove-Json -j $VarJs -p "dev-tools" } else { if ([version]$offline -ge [version]"1.2.35.663") { # Create a copy of 'dev-tools' $newDevTools = $webjson.VariousJs.'dev-tools'.PSObject.Copy() # Delete the first item and change the version $newDevTools.match = $newDevTools.match[0], $newDevTools.match[2] $newDevTools.replace = $newDevTools.replace[0], $newDevTools.replace[2] $newDevTools.version.fr = '1.2.35' # Assign a copy of 'devtools' to the 'devtools' property in $web json.others $webjson.others | Add-Member -Name 'dev-tools' -Value $newDevTools -MemberType NoteProperty # leave only first item in $web json.Various Js.'devtools' match & replace $webjson.VariousJs.'dev-tools'.match = $webjson.VariousJs.'dev-tools'.match[1] $webjson.VariousJs.'dev-tools'.replace = $webjson.VariousJs.'dev-tools'.replace[1] } } if ($urlform_goofy -and $idbox_goofy) { $webjson.VariousJs.goofyhistory.replace = $webjson.VariousJs.goofyhistory.replace -f "`"$urlform_goofy`"", "`"$idbox_goofy`"" } else { Remove-Json -j $VarJs -p "goofyhistory" } if (!($ru)) { Remove-Json -j $VarJs -p "offrujs" } if (!($premium) -or ($cache_limit)) { if (!($premium)) { $adds += $webjson.VariousJs.product_state.add } if ($cache_limit) { if ($cache_limit -lt 500) { $cache_limit = 500 } if ($cache_limit -gt 20000) { $cache_limit = 20000 } $adds2 = $webjson.VariousJs.product_state.add2 if (!([string]::IsNullOrEmpty($adds))) { $adds2 = ',' + $adds2 } $adds += $adds2 -f $cache_limit } $repl = $webjson.VariousJs.product_state.replace $webjson.VariousJs.product_state.replace = $repl -f "{pairs:{$adds}}" } else { Remove-Json -j $VarJs -p 'product_state' } if ($podcast_off -or $adsections_off) { $type = switch ($true) { { $podcast_off -and $adsections_off } { "all" } { $podcast_off -and -not $adsections_off } { "podcast" } { -not $podcast_off -and $adsections_off } { "section" } } $webjson.VariousJs.block_section.replace = $webjson.VariousJs.block_section.replace -f $type } else { Remove-Json -j $VarJs -p 'block_section' } $name = "patches.json.VariousJs." $n = "xpui.js" $contents = $webjson.VariousJs.psobject.properties.name $json = $webjson.VariousJs } } $paramdata = $xpui $novariable = "Didn't find variable " $offline_patch = $offline -replace '(\d+\.\d+\.\d+)(.\d+)', '$1' $contents | foreach { if ( $json.$PSItem.version.to ) { $to = [version]$json.$PSItem.version.to -ge [version]$offline_patch } else { $to = $true } if ( $json.$PSItem.version.fr ) { $fr = [version]$json.$PSItem.version.fr -le [version]$offline_patch } else { $fr = $false } $checkVer = $fr -and $to; $translate = $paramname -eq "RuTranslate" if ($checkVer -or $translate) { if ($json.$PSItem.match.Count -gt 1) { $count = $json.$PSItem.match.Count - 1 $numbers = 0 While ($numbers -le $count) { if ($paramdata -match $json.$PSItem.match[$numbers]) { $paramdata = $paramdata -replace $json.$PSItem.match[$numbers], $json.$PSItem.replace[$numbers] } else { $notlog = "MinJs", "MinJson", "Removertl", "RemovertlCssmin" if ($paramname -notin $notlog) { Write-Host $novariable -ForegroundColor red -NoNewline Write-Host "$name$PSItem $numbers"'in'$n } } $numbers++ } } if ($json.$PSItem.match.Count -eq 1) { if ($paramdata -match $json.$PSItem.match) { $paramdata = $paramdata -replace $json.$PSItem.match, $json.$PSItem.replace } else { if (!($translate) -or $err_ru) { Write-Host $novariable -ForegroundColor red -NoNewline Write-Host "$name$PSItem"'in'$n } } } } } $paramdata } function extract ($counts, $method, $name, $helper, $add, $patch) { switch ( $counts ) { "one" { if ($method -eq "zip") { Add-Type -Assembly 'System.IO.Compression.FileSystem' $xpui_spa_patch = Join-Path (Join-Path $env:APPDATA 'Spotify\Apps') 'xpui.spa' $zip = [System.IO.Compression.ZipFile]::Open($xpui_spa_patch, 'update') $file = $zip.GetEntry($name) $reader = New-Object System.IO.StreamReader($file.Open()) } if ($method -eq "nonezip") { $file = get-item $env:APPDATA\Spotify\Apps\xpui\$name $reader = New-Object -TypeName System.IO.StreamReader -ArgumentList $file } $xpui = $reader.ReadToEnd() $reader.Close() if ($helper) { $xpui = Helper -paramname $helper } if ($method -eq "zip") { $writer = New-Object System.IO.StreamWriter($file.Open()) } if ($method -eq "nonezip") { $writer = New-Object System.IO.StreamWriter -ArgumentList $file } $writer.BaseStream.SetLength(0) $writer.Write($xpui) if ($add) { $add | foreach { $writer.Write([System.Environment]::NewLine + $PSItem ) } } $writer.Close() if ($method -eq "zip") { $zip.Dispose() } } "more" { Add-Type -Assembly 'System.IO.Compression.FileSystem' $xpui_spa_patch = Join-Path (Join-Path $env:APPDATA 'Spotify\Apps') 'xpui.spa' $zip = [System.IO.Compression.ZipFile]::Open($xpui_spa_patch, 'update') $zip.Entries | Where-Object { $_.FullName -like $name -and $_.FullName.Split('/') -notcontains 'spotx-helper' } | foreach { $reader = New-Object System.IO.StreamReader($_.Open()) $xpui = $reader.ReadToEnd() $reader.Close() $xpui = Helper -paramname $helper $writer = New-Object System.IO.StreamWriter($_.Open()) $writer.BaseStream.SetLength(0) $writer.Write($xpui) $writer.Close() } $zip.Dispose() } "exe" { $ANSI = [Text.Encoding]::GetEncoding(1251) $xpui = [IO.File]::ReadAllText($spotifyExecutable, $ANSI) $xpui = Helper -paramname $helper [IO.File]::WriteAllText($spotifyExecutable, $xpui, $ANSI) } } } function injection { param( [Alias("p")] [string]$ArchivePath, [Alias("f")] [string]$FolderInArchive, [Alias("n")] [string]$FileName, [Alias("c")] [string]$FileContent ) $folderPathInArchive = "$($FolderInArchive)/" Add-Type -AssemblyName System.IO.Compression.FileSystem $archive = [System.IO.Compression.ZipFile]::Open($ArchivePath, 'Update') $stream = $null try { $entry = $archive.GetEntry($folderPathInArchive + $FileName) if ($entry -eq $null) { $stream = $archive.CreateEntry($folderPathInArchive + $FileName).Open() } else { $stream = $entry.Open() } $writer = [System.IO.StreamWriter]::new($stream) $writer.Write($FileContent) $writer.Dispose() $stream.Dispose() $indexEntry = $archive.Entries | Where-Object { $_.FullName -eq "index.html" } if ($indexEntry -ne $null) { $indexStream = $indexEntry.Open() $reader = [System.IO.StreamReader]::new($indexStream) $indexContent = $reader.ReadToEnd() $reader.Dispose() $indexStream.Dispose() $scriptTagIndex = $indexContent.IndexOf("") $indexEntry.Delete() $newIndexEntry = $archive.CreateEntry("index.html").Open() $indexWriter = [System.IO.StreamWriter]::new($newIndexEntry) $indexWriter.Write($modifiedIndexContent) $indexWriter.Dispose() $newIndexEntry.Dispose() } else { Write-Warning "