ShellServer.psm1
# # Set port to use. It shouldn't have collisions as Udp, but user might want to change it. # # if you change the port number here, you must change below too $Port = 5432 Register-EngineEvent PowerShell.Exiting -action { $Sock = New-Object System.Net.Sockets.UdpClient # Change port number below if you changed above. Don't use variables. $Sock.Connect('127.0.0.1', 5432) $Sock.Send([System.Text.Encoding]::UTF8.GetBytes('2Exit')) } > $nul # # Create socket objects to interact with server # $LocalHost = '127.0.0.1' $Encoder = [System.Text.Encoding]::UTF8 $Sock = New-Object System.Net.Sockets.UdpClient $Sock.Client.ReceiveTimeout = 3000 $Sock.Connect($LocalHost, $Port) $Address = [System.Net.IpAddress]::Parse($LocalHost) $End = New-Object System.Net.IPEndPoint $Address, $Port $Buffer = $Encoder.GetBytes('2Initpwsh') $Sock.Send($Buffer) > $nul # # Define functions that will interact with server # function global:prompt { $origDollarQuestion = $global:? $origLastExitCode = $global:LASTEXITCODE # if venv > 0: venv == venv.length $venv = $env:VIRTUAL_ENV if ($venv) { $venv = ($venv.Substring($venv.LastIndexOf('\')+1)).Length + 3 } else { Write-Host '' $venv = [int] [bool] $venv } $width = $Host.UI.RawUI.BufferSize.Width - $venv $duration = (Get-History -Count 1).Duration if (-Not $duration) { $duration = 0.0 } else { $duration = [string]$duration.TotalSeconds } $Buffer = $Encoder.GetBytes('1' + [int] $origDollarQuestion + (Get-Location).Path + ";$width;$duration") $Sock.Send($Buffer) > $nul try {$response = receiver} catch { Write-Host "Server didn't respond in time." function global:prompt { "$(Get-Location)> " } Set-PSReadLineKeyHandler -Key Enter -ScriptBlock { [Microsoft.PowerShell.PSConsoleReadLine]::AcceptLine() } return "$(Get-Location)> " } $ExecutionContext.InvokeCommand.ExpandString($response) $global:LASTEXITCODE = $origLastExitCode if ($global:? -ne $origDollarQuestion) { if ($origDollarQuestion) { 1+1 } else { Write-Error '' -ErrorAction 'Ignore' } } } function p { param($arg) if (-not $arg) { Set-Location } else { if ($args) { $arg += ' ' + $args -join ' ' } $Buffer = $Encoder.GetBytes("3$arg") $Sock.Send($Buffer) > $nul $response = receiver if ($response) { Set-Location $response } else { Write-Output "ShellServer: No match found." } } } function pz { $Buffer = $Encoder.GetBytes('4') $Sock.Send($Buffer) > $nul $response = receiver $query = $args -Join ' ' if ($query) { $answer = Write-Output $response | fzf --height=~20 --layout=reverse -q $query } else { $answer = Write-Output $response | fzf --height=~20 --layout=reverse } $Sock.Send($Encoder.GetBytes('4' + $answer)) > $nul Set-Location $answer } function pls { param($opts, $path) if ($path) { try {$resPath = (Resolve-Path $path -ErrorAction 'Stop').Path} catch {$resPath = (Get-Error).TargetObject} } else { $resPath = (Get-Location).Path } $Buffer = $Encoder.GetBytes("5$opts;$resPath") $Sock.Send($Buffer) > $nul $response = receiver $ExecutionContext.InvokeCommand.ExpandString($response) } function ll { pls '-cil' ($args -join ' ') } function la { pls '-acil' ($args -join ' ') } function Switch-Theme { [CmdletBinding()] param( [Parameter()] [ArgumentCompletions('all', 'terminal', 'system', 'blue')] $arg ) $Buffer = $Encoder.GetBytes("6$arg") $Sock.Send($Buffer) > $nul Start-Sleep .1 if (($arg -eq 'terminal') -Or (-Not $arg)) { PSThemeChange } } function Search-History { $width = $Host.UI.RawUI.BufferSize.Width $height = $Host.UI.RawUI.BufferSize.Height $arg = $args -join ' ' $Buffer = $Encoder.GetBytes("7$arg;$width;$height") $Sock.Send($Buffer) > $nul $response = receiver $ExecutionContext.InvokeCommand.ExpandString($response) } function receiver { $answer = '' $iterativeAnswer = $Encoder.GetString($Sock.Receive([ref] $End)) while ($iterativeAnswer[0] -eq '1') { $answer += $iterativeAnswer.Substring(1) $iterativeAnswer = $Encoder.GetString($Sock.Receive([ref] $End)) } $answer += $iterativeAnswer.Substring(1) return $answer } # # Misc. Front-end only functions, etc. # $firstEnterKeyPress = 1 $lastCmdId = (Get-History -Count 1).Id Set-PSReadLineKeyHandler -Key Enter -ScriptBlock { $line = $cursor = $null [Microsoft.PowerShell.PSConsoleReadLine]::GetBufferState([ref]$line, [ref]$cursor) $venv = [int](Test-Path 'ENV:VIRTUAL_ENV') $currentCmdId = (Get-History -Count 1).Id if (($currentCmdId -ne $lastCmdId) -or (-not $line) -or ($firstEnterKeyPress)) { $Script:lastCmdId = $currentCmdId # will treat the enter after an empty line as the first enter key press $Script:firstEnterKeyPress = [int](-not $line) [Console]::SetCursorPosition(0, [Console]::GetCursorPosition().Item2 - (2 - $venv)) Write-Host -NoNewline "`e[J`e[34m❯`e[0m $line" } [Microsoft.PowerShell.PSConsoleReadLine]::AcceptLine() } $LightThemeColors = @{ Command = "DarkYellow" Comment = "DarkGreen" ContinuationPrompt = "DarkGray" Default = "DarkGray" Emphasis = "DarkBlue" InlinePrediction = "DarkGray" Keyword = "Green" ListPrediction = "DarkYellow" ListPredictionSelected = "`e[30;5;238m" Member = "Black" Number = "Black" Operator = "DarkGray" Parameter = "DarkGray" Selection = "`e[30;5;238m" String = "DarkCyan" Type = "Black" Variable = "Green" } $DefaultThemeColors = @{ Command = "`e[93m" Comment = "`e[32m" ContinuationPrompt = "`e[37m" Default = "`e[37m" Emphasis = "`e[96m" InlinePrediction = "`e[38;5;238m" Keyword = "`e[92m" ListPrediction = "`e[33m" ListPredictionSelected = "`e[48;5;238m" Member = "`e[97m" Number = "`e[97m" Operator = "`e[90m" Parameter = "`e[90m" Selection = "`e[30;47m" String = "`e[36m" Type = "`e[37m" Variable = "`e[92m" } $SelectedLightTheme = 0 function FirstThemeCheck { if ((get-ItemProperty -Path HKCU:\SOFTWARE\Microsoft\Windows\CurrentVersion\Themes\Personalize).SystemUsesLightTheme) { Set-PSReadLineOption -Colors $LightThemeColors $Script:SelectedLightTheme = 1 } else { Set-PSReadLineOption -Colors $DefaultThemeColors $Script:SelectedLightTheme = 0 } } FirstThemeCheck function PSThemeChange { if ($SelectedLightTheme -eq 0){ Set-PSReadLineOption -Colors $LightThemeColors $Script:SelectedLightTheme = 1 } else { Set-PSReadLineOption -Colors $DefaultThemeColors $Script:SelectedLightTheme = 0 } } # # The server will respond the first communication with the completions # for the 'p' function. It's safer to keep this in the end of file. # $raw = receiver $completions = @($raw -split ';') $scriptBlock = { param($commandName, $parameterName, $wordToComplete, $commandAst, $fakeBoundParameters) $completions | Where-Object { $_ -like "$wordToComplete*" } | ForEach-Object { $_ } } Register-ArgumentCompleter -CommandName p -ParameterName arg -ScriptBlock $scriptBlock Export-ModuleMember -Function @("p", "pz", "pls", "ll", "la", "Switch-Theme", "Search-History") |