398 lines
9.7 KiB
Go
398 lines
9.7 KiB
Go
|
//go:build windows
|
||
|
// +build windows
|
||
|
|
||
|
package tty
|
||
|
|
||
|
import (
|
||
|
"context"
|
||
|
"errors"
|
||
|
"os"
|
||
|
"syscall"
|
||
|
"unsafe"
|
||
|
|
||
|
"github.com/mattn/go-isatty"
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
rightAltPressed = 1
|
||
|
leftAltPressed = 2
|
||
|
rightCtrlPressed = 4
|
||
|
leftCtrlPressed = 8
|
||
|
shiftPressed = 0x0010
|
||
|
ctrlPressed = rightCtrlPressed | leftCtrlPressed
|
||
|
altPressed = rightAltPressed | leftAltPressed
|
||
|
)
|
||
|
|
||
|
const (
|
||
|
enableProcessedInput = 0x1
|
||
|
enableLineInput = 0x2
|
||
|
enableEchoInput = 0x4
|
||
|
enableWindowInput = 0x8
|
||
|
enableMouseInput = 0x10
|
||
|
enableInsertMode = 0x20
|
||
|
enableQuickEditMode = 0x40
|
||
|
enableExtendedFlag = 0x80
|
||
|
|
||
|
enableProcessedOutput = 1
|
||
|
enableWrapAtEolOutput = 2
|
||
|
|
||
|
keyEvent = 0x1
|
||
|
mouseEvent = 0x2
|
||
|
windowBufferSizeEvent = 0x4
|
||
|
)
|
||
|
|
||
|
var kernel32 = syscall.NewLazyDLL("kernel32.dll")
|
||
|
|
||
|
var (
|
||
|
procAllocConsole = kernel32.NewProc("AllocConsole")
|
||
|
procSetStdHandle = kernel32.NewProc("SetStdHandle")
|
||
|
procGetStdHandle = kernel32.NewProc("GetStdHandle")
|
||
|
procSetConsoleScreenBufferSize = kernel32.NewProc("SetConsoleScreenBufferSize")
|
||
|
procCreateConsoleScreenBuffer = kernel32.NewProc("CreateConsoleScreenBuffer")
|
||
|
procGetConsoleScreenBufferInfo = kernel32.NewProc("GetConsoleScreenBufferInfo")
|
||
|
procWriteConsoleOutputCharacter = kernel32.NewProc("WriteConsoleOutputCharacterW")
|
||
|
procWriteConsoleOutputAttribute = kernel32.NewProc("WriteConsoleOutputAttribute")
|
||
|
procGetConsoleCursorInfo = kernel32.NewProc("GetConsoleCursorInfo")
|
||
|
procSetConsoleCursorInfo = kernel32.NewProc("SetConsoleCursorInfo")
|
||
|
procSetConsoleCursorPosition = kernel32.NewProc("SetConsoleCursorPosition")
|
||
|
procReadConsoleInput = kernel32.NewProc("ReadConsoleInputW")
|
||
|
procGetConsoleMode = kernel32.NewProc("GetConsoleMode")
|
||
|
procSetConsoleMode = kernel32.NewProc("SetConsoleMode")
|
||
|
procFillConsoleOutputCharacter = kernel32.NewProc("FillConsoleOutputCharacterW")
|
||
|
procFillConsoleOutputAttribute = kernel32.NewProc("FillConsoleOutputAttribute")
|
||
|
procScrollConsoleScreenBuffer = kernel32.NewProc("ScrollConsoleScreenBufferW")
|
||
|
)
|
||
|
|
||
|
type wchar uint16
|
||
|
type short int16
|
||
|
type dword uint32
|
||
|
type word uint16
|
||
|
|
||
|
type coord struct {
|
||
|
x short
|
||
|
y short
|
||
|
}
|
||
|
|
||
|
type smallRect struct {
|
||
|
left short
|
||
|
top short
|
||
|
right short
|
||
|
bottom short
|
||
|
}
|
||
|
|
||
|
type consoleScreenBufferInfo struct {
|
||
|
size coord
|
||
|
cursorPosition coord
|
||
|
attributes word
|
||
|
window smallRect
|
||
|
maximumWindowSize coord
|
||
|
}
|
||
|
|
||
|
type consoleCursorInfo struct {
|
||
|
size dword
|
||
|
visible int32
|
||
|
}
|
||
|
|
||
|
type inputRecord struct {
|
||
|
eventType word
|
||
|
_ [2]byte
|
||
|
event [16]byte
|
||
|
}
|
||
|
|
||
|
type keyEventRecord struct {
|
||
|
keyDown int32
|
||
|
repeatCount word
|
||
|
virtualKeyCode word
|
||
|
virtualScanCode word
|
||
|
unicodeChar wchar
|
||
|
controlKeyState dword
|
||
|
}
|
||
|
|
||
|
type windowBufferSizeRecord struct {
|
||
|
size coord
|
||
|
}
|
||
|
|
||
|
type mouseEventRecord struct {
|
||
|
mousePos coord
|
||
|
buttonState dword
|
||
|
controlKeyState dword
|
||
|
eventFlags dword
|
||
|
}
|
||
|
|
||
|
type charInfo struct {
|
||
|
unicodeChar wchar
|
||
|
attributes word
|
||
|
}
|
||
|
|
||
|
type TTY struct {
|
||
|
in *os.File
|
||
|
out *os.File
|
||
|
st uint32
|
||
|
rs []rune
|
||
|
ws chan WINSIZE
|
||
|
sigwinchCtx context.Context
|
||
|
sigwinchCtxCancel context.CancelFunc
|
||
|
readNextKeyUp bool
|
||
|
}
|
||
|
|
||
|
func readConsoleInput(fd uintptr, record *inputRecord) (err error) {
|
||
|
var w uint32
|
||
|
r1, _, err := procReadConsoleInput.Call(fd, uintptr(unsafe.Pointer(record)), 1, uintptr(unsafe.Pointer(&w)))
|
||
|
if r1 == 0 {
|
||
|
return err
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func open(path string) (*TTY, error) {
|
||
|
tty := new(TTY)
|
||
|
if false && isatty.IsTerminal(os.Stdin.Fd()) {
|
||
|
tty.in = os.Stdin
|
||
|
} else {
|
||
|
in, err := syscall.Open("CONIN$", syscall.O_RDWR, 0)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
tty.in = os.NewFile(uintptr(in), "/dev/tty")
|
||
|
}
|
||
|
|
||
|
if isatty.IsTerminal(os.Stdout.Fd()) {
|
||
|
tty.out = os.Stdout
|
||
|
} else {
|
||
|
procAllocConsole.Call()
|
||
|
out, err := syscall.Open("CONOUT$", syscall.O_RDWR, 0)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
tty.out = os.NewFile(uintptr(out), "/dev/tty")
|
||
|
}
|
||
|
|
||
|
h := tty.in.Fd()
|
||
|
var st uint32
|
||
|
r1, _, err := procGetConsoleMode.Call(h, uintptr(unsafe.Pointer(&st)))
|
||
|
if r1 == 0 {
|
||
|
return nil, err
|
||
|
}
|
||
|
tty.st = st
|
||
|
|
||
|
st &^= enableEchoInput
|
||
|
st &^= enableInsertMode
|
||
|
st &^= enableLineInput
|
||
|
st &^= enableMouseInput
|
||
|
st &^= enableWindowInput
|
||
|
st &^= enableExtendedFlag
|
||
|
st &^= enableQuickEditMode
|
||
|
|
||
|
// ignore error
|
||
|
procSetConsoleMode.Call(h, uintptr(st))
|
||
|
|
||
|
tty.ws = make(chan WINSIZE)
|
||
|
tty.sigwinchCtx, tty.sigwinchCtxCancel = context.WithCancel(context.Background())
|
||
|
|
||
|
return tty, nil
|
||
|
}
|
||
|
|
||
|
func (tty *TTY) buffered() bool {
|
||
|
return len(tty.rs) > 0
|
||
|
}
|
||
|
|
||
|
func (tty *TTY) readRune() (rune, error) {
|
||
|
if len(tty.rs) > 0 {
|
||
|
r := tty.rs[0]
|
||
|
tty.rs = tty.rs[1:]
|
||
|
return r, nil
|
||
|
}
|
||
|
var ir inputRecord
|
||
|
err := readConsoleInput(tty.in.Fd(), &ir)
|
||
|
if err != nil {
|
||
|
return 0, err
|
||
|
}
|
||
|
|
||
|
switch ir.eventType {
|
||
|
case windowBufferSizeEvent:
|
||
|
wr := (*windowBufferSizeRecord)(unsafe.Pointer(&ir.event))
|
||
|
ws := WINSIZE{
|
||
|
W: int(wr.size.x),
|
||
|
H: int(wr.size.y),
|
||
|
}
|
||
|
|
||
|
if err := tty.sigwinchCtx.Err(); err != nil {
|
||
|
// closing
|
||
|
// the following select might panic without this guard close
|
||
|
return 0, err
|
||
|
}
|
||
|
|
||
|
select {
|
||
|
case tty.ws <- ws:
|
||
|
case <-tty.sigwinchCtx.Done():
|
||
|
return 0, tty.sigwinchCtx.Err()
|
||
|
default:
|
||
|
return 0, nil // no one is currently trying to read
|
||
|
}
|
||
|
case keyEvent:
|
||
|
kr := (*keyEventRecord)(unsafe.Pointer(&ir.event))
|
||
|
if kr.keyDown == 0 {
|
||
|
if kr.unicodeChar != 0 && tty.readNextKeyUp {
|
||
|
tty.readNextKeyUp = false
|
||
|
if 0x2000 <= kr.unicodeChar && kr.unicodeChar < 0x3000 {
|
||
|
return rune(kr.unicodeChar), nil
|
||
|
}
|
||
|
}
|
||
|
} else {
|
||
|
if kr.controlKeyState&altPressed != 0 && kr.unicodeChar > 0 {
|
||
|
tty.rs = []rune{rune(kr.unicodeChar)}
|
||
|
return rune(0x1b), nil
|
||
|
}
|
||
|
if kr.unicodeChar > 0 {
|
||
|
if kr.controlKeyState&shiftPressed != 0 {
|
||
|
switch kr.unicodeChar {
|
||
|
case 0x09:
|
||
|
tty.rs = []rune{0x5b, 0x5a}
|
||
|
return rune(0x1b), nil
|
||
|
}
|
||
|
}
|
||
|
return rune(kr.unicodeChar), nil
|
||
|
}
|
||
|
vk := kr.virtualKeyCode
|
||
|
if kr.controlKeyState&ctrlPressed != 0 {
|
||
|
switch vk {
|
||
|
case 0x21: // ctrl-page-up
|
||
|
tty.rs = []rune{0x5b, 0x35, 0x3B, 0x35, 0x7e}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x22: // ctrl-page-down
|
||
|
tty.rs = []rune{0x5b, 0x36, 0x3B, 0x35, 0x7e}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x23: // ctrl-end
|
||
|
tty.rs = []rune{0x5b, 0x31, 0x3B, 0x35, 0x46}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x24: // ctrl-home
|
||
|
tty.rs = []rune{0x5b, 0x31, 0x3B, 0x35, 0x48}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x25: // ctrl-left
|
||
|
tty.rs = []rune{0x5b, 0x31, 0x3B, 0x35, 0x44}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x26: // ctrl-up
|
||
|
tty.rs = []rune{0x5b, 0x31, 0x3B, 0x35, 0x41}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x27: // ctrl-right
|
||
|
tty.rs = []rune{0x5b, 0x31, 0x3B, 0x35, 0x43}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x28: // ctrl-down
|
||
|
tty.rs = []rune{0x5b, 0x31, 0x3B, 0x35, 0x42}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x2e: // ctrl-delete
|
||
|
tty.rs = []rune{0x5b, 0x33, 0x3B, 0x35, 0x7e}
|
||
|
return rune(0x1b), nil
|
||
|
}
|
||
|
}
|
||
|
switch vk {
|
||
|
case 0x12: // menu
|
||
|
if kr.controlKeyState&leftAltPressed != 0 {
|
||
|
tty.readNextKeyUp = true
|
||
|
}
|
||
|
return 0, nil
|
||
|
case 0x21: // page-up
|
||
|
tty.rs = []rune{0x5b, 0x35, 0x7e}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x22: // page-down
|
||
|
tty.rs = []rune{0x5b, 0x36, 0x7e}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x23: // end
|
||
|
tty.rs = []rune{0x5b, 0x46}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x24: // home
|
||
|
tty.rs = []rune{0x5b, 0x48}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x25: // left
|
||
|
tty.rs = []rune{0x5b, 0x44}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x26: // up
|
||
|
tty.rs = []rune{0x5b, 0x41}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x27: // right
|
||
|
tty.rs = []rune{0x5b, 0x43}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x28: // down
|
||
|
tty.rs = []rune{0x5b, 0x42}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x2e: // delete
|
||
|
tty.rs = []rune{0x5b, 0x33, 0x7e}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x70, 0x71, 0x72, 0x73: // F1,F2,F3,F4
|
||
|
tty.rs = []rune{0x5b, 0x4f, rune(vk) - 0x20}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x074, 0x75, 0x76, 0x77: // F5,F6,F7,F8
|
||
|
tty.rs = []rune{0x5b, 0x31, rune(vk) - 0x3f, 0x7e}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x78, 0x79: // F9,F10
|
||
|
tty.rs = []rune{0x5b, 0x32, rune(vk) - 0x48, 0x7e}
|
||
|
return rune(0x1b), nil
|
||
|
case 0x7a, 0x7b: // F11,F12
|
||
|
tty.rs = []rune{0x5b, 0x32, rune(vk) - 0x47, 0x7e}
|
||
|
return rune(0x1b), nil
|
||
|
}
|
||
|
return 0, nil
|
||
|
}
|
||
|
}
|
||
|
return 0, nil
|
||
|
}
|
||
|
|
||
|
func (tty *TTY) close() error {
|
||
|
procSetConsoleMode.Call(tty.in.Fd(), uintptr(tty.st))
|
||
|
tty.sigwinchCtxCancel()
|
||
|
close(tty.ws)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func (tty *TTY) size() (int, int, error) {
|
||
|
var csbi consoleScreenBufferInfo
|
||
|
r1, _, err := procGetConsoleScreenBufferInfo.Call(tty.out.Fd(), uintptr(unsafe.Pointer(&csbi)))
|
||
|
if r1 == 0 {
|
||
|
return 0, 0, err
|
||
|
}
|
||
|
return int(csbi.window.right - csbi.window.left + 1), int(csbi.window.bottom - csbi.window.top + 1), nil
|
||
|
}
|
||
|
|
||
|
func (tty *TTY) sizePixel() (int, int, int, int, error) {
|
||
|
x, y, err := tty.size()
|
||
|
if err != nil {
|
||
|
x = -1
|
||
|
y = -1
|
||
|
}
|
||
|
return x, y, -1, -1, errors.New("no implemented method for querying size in pixels on Windows")
|
||
|
}
|
||
|
|
||
|
func (tty *TTY) input() *os.File {
|
||
|
return tty.in
|
||
|
}
|
||
|
|
||
|
func (tty *TTY) output() *os.File {
|
||
|
return tty.out
|
||
|
}
|
||
|
|
||
|
func (tty *TTY) raw() (func() error, error) {
|
||
|
var st uint32
|
||
|
r1, _, err := procGetConsoleMode.Call(tty.in.Fd(), uintptr(unsafe.Pointer(&st)))
|
||
|
if r1 == 0 {
|
||
|
return nil, err
|
||
|
}
|
||
|
mode := st &^ (enableEchoInput | enableProcessedInput | enableLineInput | enableProcessedOutput)
|
||
|
r1, _, err = procSetConsoleMode.Call(tty.in.Fd(), uintptr(mode))
|
||
|
if r1 == 0 {
|
||
|
return nil, err
|
||
|
}
|
||
|
return func() error {
|
||
|
r1, _, err := procSetConsoleMode.Call(tty.in.Fd(), uintptr(st))
|
||
|
if r1 == 0 {
|
||
|
return err
|
||
|
}
|
||
|
return nil
|
||
|
}, nil
|
||
|
}
|
||
|
|
||
|
func (tty *TTY) sigwinch() <-chan WINSIZE {
|
||
|
return tty.ws
|
||
|
}
|