25

我已经搜索过,但显然我的 google foo 很弱。我需要的是一种在控制台中提示用户输入并在一段时间后使请求超时并在没有输入时继续执行脚本的方法。据我所知,Read-Host 不提供此功能功能。$host.UI.PromptForChoice() 和 $host.UI.RawUI.ReadKey() 也没有。在此先感谢您的任何指点。

编辑:非常感谢 Lars Truijens 找到了答案。我把他指出的代码封装成一个函数。请注意,我实现它的方式意味着在用户按下键和脚本继续执行之间可能会有长达一秒的延迟。

function Pause-Host
{
    param(
            $Delay = 1
         )
    $counter = 0;
    While(!$host.UI.RawUI.KeyAvailable -and ($counter++ -lt $Delay))
    {
        [Threading.Thread]::Sleep(1000)
    }
}
4

7 回答 7

19

在这里找到了一些东西:

$counter = 0
while(!$Host.UI.RawUI.KeyAvailable -and ($counter++ -lt 600))
{
      [Threading.Thread]::Sleep( 1000 )
}
于 2008-09-29T19:44:47.937 回答
6

它现在已经很老了,但是我如何基于相同的 KeyAvailable 方法解决它是在这里:

https://gist.github.com/nathanchere/704920a4a43f06f4f0d2

它等待 x 秒,.在最长等待时间之前每秒钟显示一个。如果一个键被按下它返回$true,否则$false

Function TimedPrompt($prompt,$secondsToWait){   
    Write-Host -NoNewline $prompt
    $secondsCounter = 0
    $subCounter = 0
    While ( (!$host.ui.rawui.KeyAvailable) -and ($count -lt $secondsToWait) ){
        start-sleep -m 10
        $subCounter = $subCounter + 10
        if($subCounter -eq 1000)
        {
            $secondsCounter++
            $subCounter = 0
            Write-Host -NoNewline "."
        }       
        If ($secondsCounter -eq $secondsToWait) { 
            Write-Host "`r`n"
            return $false;
        }
    }
    Write-Host "`r`n"
    return $true;
}

并使用:

$val = TimedPrompt "Press key to cancel restore; will begin in 3 seconds" 3
Write-Host $val
于 2015-03-30T10:00:48.633 回答
3

对于正在寻找现代解决方案的人来说,在预定义的按键上退出 PowerShell 脚本有额外的约束,以下解决方案可能会对您有所帮助:

Write-Host ("PowerShell Script to run a loop and exit on pressing 'q'!")
$count=0
$sleepTimer=500 #in milliseconds
$QuitKey=81 #Character code for 'q' key.
while($count -le 100)
{
    if($host.UI.RawUI.KeyAvailable) {
        $key = $host.ui.RawUI.ReadKey("NoEcho,IncludeKeyUp")
        if($key.VirtualKeyCode -eq $QuitKey) {
            #For Key Combination: eg., press 'LeftCtrl + q' to quit.
            #Use condition: (($key.VirtualKeyCode -eq $Qkey) -and ($key.ControlKeyState -match "LeftCtrlPressed"))
            Write-Host -ForegroundColor Yellow ("'q' is pressed! Stopping the script now.")
            break
        }
    }
    #Do your operations
    $count++
    Write-Host ("Count Incremented to - {0}" -f $count)
    Write-Host ("Press 'q' to stop the script!")
    Start-Sleep -m $sleepTimer
}
Write-Host -ForegroundColor Green ("The script has stopped.")

示例脚本输出: 在此处输入图像描述

请参阅有关键状态的Microsoft 文档以处理更多组合。

致谢:Technet 链接

于 2018-05-10T13:47:51.977 回答
1

这是一个接受的击键实用函数:

  • 验证字符集(作为 1 个字符的正则表达式)。
  • 可选消息
  • 可选超时时间(以秒为单位)

只有匹配的击键才会反映到屏幕上。

用法:

$key = GetKeyPress '[ynq]' "Run step X ([y]/n/q)?" 5

if ($key -eq $null)
{
    Write-Host "No key was pressed.";
}
else
{
    Write-Host "The key was '$($key)'."
}

执行:

Function GetKeyPress([string]$regexPattern='[ynq]', [string]$message=$null, [int]$timeOutSeconds=0)
{
    $key = $null

    $Host.UI.RawUI.FlushInputBuffer() 

    if (![string]::IsNullOrEmpty($message))
    {
        Write-Host -NoNewLine $message
    }

    $counter = $timeOutSeconds * 1000 / 250
    while($key -eq $null -and ($timeOutSeconds -eq 0 -or $counter-- -gt 0))
    {
        if (($timeOutSeconds -eq 0) -or $Host.UI.RawUI.KeyAvailable)
        {                       
            $key_ = $host.UI.RawUI.ReadKey("NoEcho,IncludeKeyDown,IncludeKeyUp")
            if ($key_.KeyDown -and $key_.Character -match $regexPattern)
            {
                $key = $key_                    
            }
        }
        else
        {
            Start-Sleep -m 250  # Milliseconds
        }
    }                       

    if (-not ($key -eq $null))
    {
        Write-Host -NoNewLine "$($key.Character)" 
    }

    if (![string]::IsNullOrEmpty($message))
    {
        Write-Host "" # newline
    }       

    return $(if ($key -eq $null) {$null} else {$key.Character})
}
于 2018-09-27T23:46:08.627 回答
0
function ReadKeyWithDefault($prompt, $defaultKey, [int]$timeoutInSecond = 5 ) {
    $counter =  $timeoutInSecond * 10
    do{
        $remainingSeconds = [math]::floor($counter / 10)
        Write-Host "`r$prompt (default  $defaultKey  in $remainingSeconds seconds): " -NoNewline
        if($Host.UI.RawUI.KeyAvailable){
            $key = $host.UI.RawUI.ReadKey("IncludeKeyUp")
            Write-Host 
            return $key
        }
        Start-Sleep -Milliseconds 100
    }while($counter-- -gt 0)

    Write-Host $defaultKey
    return $defaultKey
}

$readKey = ReadKeyWithDefault "If error auto exit( y/n )" 'y' 5
于 2021-03-17T07:45:48.273 回答
0
timeout.exe 5

仍然可以从 powershell 工作。等待 5 秒或直到按键。

也许不优雅。但是很容易。

然而,

从 Powershell_ISE 它会弹出一个新的命令提示符窗口,并立即返回,因此它不会等待(从 powershell 控制台它使用该控制台并确实等待)。您可以通过更多的工作让它从 ISE 等待(仍然会弹出它自己的窗口):

if ($psISE) {
    start -Wait timeout.exe 5
} else {
    timeout.exe 5
}
于 2021-07-17T14:00:58.977 回答
0

另一种选择:您可以使用choice自 Windows 2000 以来每个 Windows 版本附带的 shell 命令

Choice /C yn /D n /t 5 /m "Are you sure? You have 5 seconds to decide"
if ($LASTEXITCODE -eq "1") # 1 for "yes" 2 for "no"
{
    # do stuff
}
else
{
    # don't do stuff
}

Stackoverflow 语法高亮不适用于 powershell,#在此处表示“注释”

于 2021-09-06T09:16:31.643 回答