2021-01-25 12:35:32 -08:00
|
|
|
<?php
|
|
|
|
|
|
|
|
declare(strict_types=1);
|
2020-12-22 14:19:32 -08:00
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (c) 2020 https://rewiv.com sikofitt@gmail.com
|
|
|
|
*
|
|
|
|
* This Source Code Form is subject to the
|
|
|
|
* terms of the Mozilla Public License, v. 2.0.
|
|
|
|
*
|
|
|
|
* If a copy of the MPL was not distributed with this file,
|
|
|
|
* You can obtain one at https://mozilla.org/MPL/2.0/.
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace Sikofitt\Console;
|
|
|
|
|
|
|
|
use FFI;
|
|
|
|
use RuntimeException;
|
|
|
|
|
|
|
|
final class Getch
|
|
|
|
{
|
2021-02-15 11:44:39 -08:00
|
|
|
// Special key codes
|
|
|
|
public const GETCH_SPECIAL = 0;
|
|
|
|
public const GETCH_F1 = 59;
|
|
|
|
public const GETCH_F2 = 60;
|
|
|
|
public const GETCH_F3 = 61;
|
|
|
|
public const GETCH_F4 = 62;
|
|
|
|
public const GETCH_F5 = 63;
|
|
|
|
public const GETCH_F6 = 64;
|
|
|
|
public const GETCH_F7 = 65;
|
|
|
|
public const GETCH_F8 = 66;
|
|
|
|
public const GETCH_F9 = 67;
|
|
|
|
public const GETCH_F10 = 68;
|
|
|
|
public const GETCH_F11 = 87;
|
|
|
|
public const GETCH_F12 = 88;
|
|
|
|
public const GETCH_UP_ARROW = 72;
|
|
|
|
public const GETCH_LEFT_ARROW = 75;
|
|
|
|
public const GETCH_RIGHT_ARROW = 77;
|
|
|
|
public const GETCH_DOWN_ARROW = 80;
|
|
|
|
public const GETCH_DELETE = 83;
|
|
|
|
public const GETCH_HOME = 102;
|
|
|
|
public const GETCH_PGUP = 104;
|
|
|
|
public const GETCH_END = 107;
|
|
|
|
public const GETCH_PGDN = 109;
|
|
|
|
public const GETCH_INSERT = 110;
|
|
|
|
|
2021-01-25 12:35:32 -08:00
|
|
|
private const LINUX_LIBRARY = __DIR__.'/Resources/libgetch.so';
|
2020-12-22 14:19:32 -08:00
|
|
|
private const WINDOWS_LIBRARY = 'ucrtbase.dll';
|
2021-01-25 12:35:32 -08:00
|
|
|
private const DECLARATIONS = <<<DECLARATIONS
|
|
|
|
int _getch();
|
|
|
|
int _ungetch(int c);
|
|
|
|
DECLARATIONS;
|
2020-12-22 14:19:32 -08:00
|
|
|
|
|
|
|
private static ?FFI $ffi = null;
|
|
|
|
|
2021-02-15 11:44:39 -08:00
|
|
|
public static function resetFFI(): void
|
|
|
|
{
|
|
|
|
static::$ffi = null;
|
|
|
|
}
|
|
|
|
|
2020-12-22 14:19:32 -08:00
|
|
|
public function __construct(string $linuxLibrary = null)
|
|
|
|
{
|
|
|
|
if (null === $linuxLibrary) {
|
|
|
|
$linuxLibrary = self::LINUX_LIBRARY;
|
|
|
|
}
|
|
|
|
|
2021-01-25 12:35:32 -08:00
|
|
|
if (null === self::$ffi) {
|
2020-12-22 14:19:32 -08:00
|
|
|
$osFamily = PHP_OS_FAMILY;
|
2021-01-25 12:35:32 -08:00
|
|
|
if ('Windows' === $osFamily) {
|
|
|
|
self::$ffi = FFI::cdef(self::DECLARATIONS, self::WINDOWS_LIBRARY);
|
|
|
|
} elseif ('Linux' === $osFamily) {
|
2020-12-22 14:19:32 -08:00
|
|
|
if (!file_exists($linuxLibrary)) {
|
|
|
|
throw new RuntimeException(sprintf('Could not find library file %s.', $linuxLibrary));
|
|
|
|
}
|
|
|
|
|
2021-01-25 12:35:32 -08:00
|
|
|
self::$ffi = FFI::cdef(self::DECLARATIONS, $linuxLibrary);
|
2020-12-22 14:19:32 -08:00
|
|
|
} else {
|
|
|
|
throw new RuntimeException(sprintf('Sorry, %s is not supported yet.', $osFamily));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-02-22 17:35:17 -08:00
|
|
|
/* public function keyCode(string $device): array
|
|
|
|
{
|
|
|
|
|
|
|
|
$arrayType = \FFI::arrayType(self::$ffi->type('int'), [3]);
|
|
|
|
$res = \FFI::new($arrayType);
|
|
|
|
|
|
|
|
$arrayType = self::$ffi->keyCode($device);
|
|
|
|
|
|
|
|
return [
|
|
|
|
'type' => $arrayType[0],
|
|
|
|
'code' => $arrayType[1],
|
|
|
|
'value' => $arrayType[2],
|
|
|
|
'keyCode' => $arrayType[3],
|
|
|
|
];
|
|
|
|
}*/
|
|
|
|
|
2021-01-25 12:35:32 -08:00
|
|
|
public function getch(): int
|
2020-12-22 14:19:32 -08:00
|
|
|
{
|
2021-01-07 13:56:08 -08:00
|
|
|
return self::$ffi->_getch();
|
|
|
|
}
|
|
|
|
|
2021-01-25 12:35:32 -08:00
|
|
|
public function ungetch($char): int
|
2021-01-07 14:13:05 -08:00
|
|
|
{
|
2021-01-25 12:35:32 -08:00
|
|
|
if (!is_string($char) && !is_int($char)) {
|
|
|
|
throw new \TypeError('ungetch takes a parameter of int or string.');
|
|
|
|
}
|
2021-01-07 14:13:05 -08:00
|
|
|
|
2021-01-25 12:35:32 -08:00
|
|
|
if (is_string($char)) {
|
|
|
|
$char = ord($char[0]);
|
2021-01-07 14:13:05 -08:00
|
|
|
}
|
|
|
|
|
2021-01-25 12:35:32 -08:00
|
|
|
return self::$ffi->_ungetch($char);
|
2020-12-22 14:19:32 -08:00
|
|
|
}
|
|
|
|
}
|