首页 > 解决方案 > 如何使用 PHP-CLI 获取光标位置?

问题描述

使用在 CLI 模式下运行的 PHP 脚本,我想以可移植的方式获取光标位置。

使用代码:

// Query Cursor Position
echo "\033[6n";

在终端中,此代码报告光标位置,如

wb ?> ./script.php 
^[[5;1R
wb ?> ;1R 

但是,我无法检索代码中的两个值(行:5,列:1)。

经过一些输出缓冲测试:

ob_start();
echo "\033[6n";
$s = ob_get_contents();
file_put_contents('cpos.txt',$s);

我在 cpos.txt 文件中有“\033[6n”,而不是设备答案。

并阅读 STDIN :

$timeout = 2;
$sent = false;
$t = microtime(true);
$buf = '';
stream_set_blocking(STDIN,false);
while(true){
    $buf .= fread(STDIN,8);
    if(!$sent){
        echo "\033[6n";
        $sent = true;
    }
    if($t+$timeout<microtime(true))
        break;
}
var_dump($buf);

缓冲区为空,但终端显示设备答案:

wb ?> ./script.php 
^[[5;1R
string(0) ""
wb ?>

有没有办法在没有诅咒的情况下获得光标位置?

标签: phpterminalcommand-line-interface

解决方案


到目前为止,您的代码几乎可以正常工作,您会发现按回车键并等待超时完成确实会生成一个包含答案的字符串,但\n末尾有一个字符。(注意字符串长度为 7 而不是 0。)

$ php foo.php
^[[2;1R                           
string(7) "
"

这里的问题是stream_set_blocking不会阻止终端逐行缓冲输入,因此在按下回车键之前,终端不会向程序的标准输入发送任何内容。

要使终端在没有行缓冲的情况下立即向您的程序发送字符,您需要将终端设置为“非规范”模式。这会禁用任何行编辑功能,例如按退格键擦除字符的能力,而是立即将字符发送到输入缓冲区。在 PHP 中执行此操作的最简单方法是调用 Unix 实用程序stty

<?php
system('stty -icanon');

echo "\033[6n";
$buf = fread(STDIN, 16);

var_dump($buf);

此代码成功地将来自终端的响应捕获到$buf.

$ php foo.php
^[[2;1Rstring(6) ""

但是,此代码有几个问题。首先,它完成后不会在终端中重新启用规范模式。当您稍后在程序中尝试从标准输入输入时,或者在程序退出后在您的 shell 中输入时,这可能会导致问题。其次,来自终端的响应代码^[[2;1R仍然回显到终端,当您只想将其读入变量时,这会使程序的输出看起来很混乱。

为了解决输入回显问题,我们可以添加参数-echostty禁用终端中的输入回显。要将终端重置为我们更改之前的状态,我们可以调用stty -g以输出当前终端设置的列表,stty稍后可以将其传递给以重置终端。

<?php
// Save terminal settings.
$ttyprops = trim(`stty -g`);

// Disable canonical input and disable echo.
system('stty -icanon -echo');

echo "\033[6n";
$buf = fread(STDIN, 16);

// Restore terminal settings.
system("stty '$ttyprops'");

var_dump($buf);

现在运行程序时,我们看不到终端中显示任何垃圾:

$ php foo.php 
string(6) ""

我们可以对此做出的最后一个潜在改进是允许程序在 stdout 被重定向到另一个进程/文件时运行。这对于您的应用程序可能是必需的,也可能不是必需的,但目前,运行php foo.php > /tmp/outfile将无法正常工作,因为echo "\033[6n";将直接写入输出文件而不是终端,使您的程序等待字符发送到标准输入,因为终端从未发送过任何转义序列,因此不会响应它。一种解决方法是写入/dev/tty而不是标准输出,如下所示:

$term = fopen('/dev/tty', 'w');
fwrite($term, "\033[6n");
fclose($term); // Flush and close the file.

将所有这些放在一起,并使用bin2hex()而不是var_dump()获取 中的字符列表$buf,我们得到以下内容:

<?php
$ttyprops = trim(`stty -g`);
system('stty -icanon -echo');

$term = fopen('/dev/tty', 'w');
fwrite($term, "\033[6n");
fclose($term);

$buf = fread(STDIN, 16);

system("stty '$ttyprops'");

echo bin2hex($buf) . "\n";

我们可以看到程序正常运行如下:

$ php foo.php > /tmp/outfile
$ cat /tmp/outfile
1b5b323b3152
$ xxd -p -r /tmp/outfile | xxd
00000000: 1b5b 323b 3152                           .[2;1R

这表明$bufcontains ^[[2;1R,表示在查询其位置时光标位于第 2 行和第 1 列。

所以现在剩下要做的就是在 PHP 中解析这个字符串并提取用分号分隔的行和列。这可以通过正则表达式来完成。

<?php
// Example response string.
$buf = "\033[123;456R";

$matches = [];
preg_match('/^\033\[(\d+);(\d+)R$/', $buf, $matches);

$row = intval($matches[1]);
$col = intval($matches[2]);
echo "Row: $row, Col: $col\n";

这给出了以下输出:

Row: 123, Col: 456

值得注意的是,所有这些代码只能移植到类 Unix 操作系统和兼容 ANSI/VT100 的终端上。除非您在 Cygwin / MSYS2 下运行该程序,否则此代码可能无法在 Windows 上运行。我还建议您在此代码中添加一些错误处理,以防无论出于何种原因您都没有从终端获得预期的响应。


推荐阅读