123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905 |
- package getch
- import (
- "fmt"
- "os"
- "os/signal"
- "strconv"
- "strings"
- "sync"
- "syscall"
- "time"
- "unicode/utf8"
- "unsafe"
- )
- func attachReader() (getch, error) {
-
- keys, err := detectTerminal()
- if err == nil {
- var inFd int
-
- inFd, err = syscall.Open("/dev/tty", syscall.O_RDONLY, 0)
- if err == nil {
- var out *os.File
-
- out, err = os.OpenFile("/dev/tty", syscall.O_WRONLY, 0)
- if err == nil {
- gl := &getchLinux{make(chan os.Signal), inFd, out, keys, syscall.Termios{},
- make(chan *internalKeyEvent), make(chan bool), &sync.WaitGroup{}}
- in := uintptr(gl.inFd)
-
-
- signal.Notify(gl.sigio, syscall.SIGIO)
-
-
-
- err = fcntl(in, syscall.F_SETFL, syscall.O_ASYNC|syscall.O_NONBLOCK)
- if err == nil {
-
-
- err = fcntl(in, syscall.F_SETOWN, syscall.Getpid())
- if err == nil {
-
- err = ioctl(gl.out.Fd(), syscall.TCGETS, &gl.origTios)
- if err == nil {
-
- tios := gl.origTios
-
-
-
-
-
-
-
-
-
- tios.Iflag &^= syscall.IGNBRK | syscall.BRKINT | syscall.PARMRK |
- syscall.ISTRIP | syscall.INLCR | syscall.IGNCR |
- syscall.ICRNL | syscall.IXON
-
-
-
-
-
-
-
- tios.Lflag &^= syscall.ECHO | syscall.ECHONL | syscall.ICANON |
- syscall.ISIG | syscall.IEXTEN
-
-
-
-
- tios.Cflag &^= syscall.CSIZE | syscall.PARENB
-
- tios.Cflag |= syscall.CS8
-
- tios.Cc[syscall.VMIN] = 1
-
- tios.Cc[syscall.VTIME] = 0
- err = ioctl(gl.out.Fd(), syscall.TCSETS, &tios)
- if err == nil {
-
- go gl.eventListener()
- return gl, nil
- }
- }
- }
- }
- }
- }
- }
- return nil, err
- }
- type getchLinux struct {
- sigio chan os.Signal
- inFd int
- out *os.File
- escKeys []string
- origTios syscall.Termios
- keyEventChan chan *internalKeyEvent
- stopChan chan bool
- wg *sync.WaitGroup
- }
- var posBuf = make([]byte, 20)
- func (gl *getchLinux) CursorPosition() (int, int, error) {
- var n, x, y int
- var err error
-
- gl.out.WriteString("\033[6n")
- if n, err = os.Stdin.Read(posBuf); n > 2 && err == nil {
- if res := string(posBuf[:n]); res[:2] == "\033[" {
-
- posStr := strings.Split(res[2:n-1], ";")
- if len(posStr) == 2 {
- if y, err = strconv.Atoi(posStr[0]); err == nil {
- x, err = strconv.Atoi(posStr[1])
- }
- }
- }
- }
- return x - 1, y, err
- }
- func (gl *getchLinux) SetCursorPosition(x, y int) error {
-
- gl.out.WriteString("\033[")
- gl.out.WriteString(fmt.Sprint(y))
- gl.out.WriteString(";")
- gl.out.WriteString(fmt.Sprint(x + 1))
- gl.out.WriteString("H")
- return nil
- }
- func (gl *getchLinux) GetKey() *internalKeyEvent {
- return <-gl.keyEventChan
- }
- func (gl *getchLinux) GetKeyAsync() *internalKeyEvent {
- select {
- case e := <-gl.keyEventChan:
- return e
- default:
- }
- return nil
- }
- func (gl *getchLinux) Close() {
-
- gl.wg.Add(1)
-
- for gl.GetKeyAsync() != nil {
- time.Sleep(10 * time.Millisecond)
- }
- gl.stopChan <- true
- gl.wg.Wait()
-
- ioctl(gl.out.Fd(), syscall.TCSETS, &gl.origTios)
-
- gl.out.Close()
- syscall.Close(gl.inFd)
- }
- func (gl *getchLinux) eventListener() {
-
- buf := make([]byte, 128)
-
- loop:
- for true {
- select {
- case <-gl.sigio:
- for {
- n, err := syscall.Read(gl.inFd, buf)
- if err == syscall.EAGAIN || err == syscall.EWOULDBLOCK {
- break
- } else if err != nil {
- gl.keyEventChan <- &internalKeyEvent{err: err}
- }
-
- select {
- case <-gl.stopChan:
- break loop
- case gl.keyEventChan <- gl.buildKeyEvent(buf[:n]):
- continue
- }
- }
- case <-gl.stopChan:
- break loop
- }
- }
- gl.wg.Done()
- }
- func (gl *getchLinux) buildKeyEvent(buf []byte) *internalKeyEvent {
- ike := &internalKeyEvent{&KeyEvent{}, nil}
- if len(buf) > 0 {
- bufstr := string(buf)
-
- if buf[0] == '\033' && len(buf) > 1 {
- for i, ek := range gl.escKeys {
- if strings.HasPrefix(bufstr, ek) {
-
- termkey := termKey(0xFFFF - i)
-
- ike.Rune = 0x00
- switch termkey {
- case termKeyF1:
- ike.Code = KeyF1
- case termKeyF2:
- ike.Code = KeyF2
- case termKeyF3:
- ike.Code = KeyF3
- case termKeyF4:
- ike.Code = KeyF4
- case termKeyF5:
- ike.Code = KeyF5
- case termKeyF6:
- ike.Code = KeyF6
- case termKeyF7:
- ike.Code = KeyF7
- case termKeyF8:
- ike.Code = KeyF8
- case termKeyF9:
- ike.Code = KeyF9
- case termKeyF10:
- ike.Code = KeyF10
- case termKeyF11:
- ike.Code = KeyF11
- case termKeyF12:
- ike.Code = KeyF12
- case termKeyInsert:
- ike.Code = KeyInsert
- case termKeyDelete:
- ike.Code = KeyDelete
- case termKeyHome:
- ike.Code = KeyHome
- case termKeyHome1:
- ike.Code = KeyHome
- case termKeyEnd:
- ike.Code = KeyEnd
- case termKeyEnd1:
- ike.Code = KeyEnd
- case termKeyPgup:
- ike.Code = KeyPgup
- case termKeyPgdn:
- ike.Code = KeyPgdn
- case termKeyArrowUp:
- ike.Code = KeyArrowUp
- case termKeyArrowDown:
- ike.Code = KeyArrowDown
- case termKeyArrowLeft:
- ike.Code = KeyArrowLeft
- case termKeyArrowRight:
- ike.Code = KeyArrowRight
- }
- }
- }
- if ike.KeyEvent.Code == "" {
-
- ike.err = &ErrUnknownEscapeSequence{buf}
- }
- } else {
-
- de, deSize := utf8.DecodeRune(buf)
- if len(buf) > deSize {
-
- ike.RawBuf = make([]byte, len(buf))
- copy(ike.RawBuf, buf)
- }
-
- ike.Rune = de
- switch de {
- case 'a':
- ike.Code = KeyA
- case 'b':
- ike.Code = KeyB
- case 'c':
- ike.Code = KeyC
- case 'd':
- ike.Code = KeyD
- case 'e':
- ike.Code = KeyE
- case 'f':
- ike.Code = KeyF
- case 'g':
- ike.Code = KeyG
- case 'h':
- ike.Code = KeyH
- case 'i':
- ike.Code = KeyI
- case 'j':
- ike.Code = KeyJ
- case 'k':
- ike.Code = KeyK
- case 'l':
- ike.Code = KeyL
- case 'm':
- ike.Code = KeyM
- case 'n':
- ike.Code = KeyN
- case 'o':
- ike.Code = KeyO
- case 'p':
- ike.Code = KeyP
- case 'q':
- ike.Code = KeyQ
- case 'r':
- ike.Code = KeyR
- case 's':
- ike.Code = KeyS
- case 't':
- ike.Code = KeyT
- case 'u':
- ike.Code = KeyU
- case 'v':
- ike.Code = KeyV
- case 'w':
- ike.Code = KeyW
- case 'x':
- ike.Code = KeyX
- case 'y':
- ike.Code = KeyY
- case 'z':
- ike.Code = KeyZ
- case '0':
- ike.Code = Key0
- case '1':
- ike.Code = Key1
- case '2':
- ike.Code = Key2
- case '3':
- ike.Code = Key3
- case '4':
- ike.Code = Key4
- case '5':
- ike.Code = Key5
- case '6':
- ike.Code = Key6
- case '7':
- ike.Code = Key7
- case '8':
- ike.Code = Key8
- case '9':
- ike.Code = Key9
-
- case 'A':
- ike.Code = KeyA
- ike.Shift = true
- case 'B':
- ike.Code = KeyB
- ike.Shift = true
- case 'C':
- ike.Code = KeyC
- ike.Shift = true
- case 'D':
- ike.Code = KeyD
- ike.Shift = true
- case 'E':
- ike.Code = KeyE
- ike.Shift = true
- case 'F':
- ike.Code = KeyF
- ike.Shift = true
- case 'G':
- ike.Code = KeyG
- ike.Shift = true
- case 'H':
- ike.Code = KeyH
- ike.Shift = true
- case 'I':
- ike.Code = KeyI
- ike.Shift = true
- case 'J':
- ike.Code = KeyJ
- ike.Shift = true
- case 'K':
- ike.Code = KeyK
- ike.Shift = true
- case 'L':
- ike.Code = KeyL
- ike.Shift = true
- case 'M':
- ike.Code = KeyM
- ike.Shift = true
- case 'N':
- ike.Code = KeyN
- ike.Shift = true
- case 'O':
- ike.Code = KeyO
- ike.Shift = true
- case 'P':
- ike.Code = KeyP
- ike.Shift = true
- case 'Q':
- ike.Code = KeyQ
- ike.Shift = true
- case 'R':
- ike.Code = KeyR
- ike.Shift = true
- case 'S':
- ike.Code = KeyS
- ike.Shift = true
- case 'T':
- ike.Code = KeyT
- ike.Shift = true
- case 'U':
- ike.Code = KeyU
- ike.Shift = true
- case 'V':
- ike.Code = KeyV
- ike.Shift = true
- case 'W':
- ike.Code = KeyW
- ike.Shift = true
- case 'X':
- ike.Code = KeyX
- ike.Shift = true
- case 'Y':
- ike.Code = KeyY
- ike.Shift = true
- case 'Z':
- ike.Code = KeyZ
- ike.Shift = true
- case 0x01:
- ike.Code = KeyA
- ike.Ctrl = true
- case 0x02:
- ike.Code = KeyB
- ike.Ctrl = true
- case 0x03:
- ike.Code = KeyC
- ike.Ctrl = true
- case 0x04:
- ike.Code = KeyD
- ike.Ctrl = true
- case 0x05:
- ike.Code = KeyE
- ike.Ctrl = true
- case 0x06:
- ike.Code = KeyF
- ike.Ctrl = true
- case 0x07:
- ike.Code = KeyG
- ike.Ctrl = true
- case 0x08:
- ike.Code = KeyH
- ike.Ctrl = true
- case 0x09:
- ike.Code = KeyI
- ike.Ctrl = true
- case 0x0A:
- ike.Code = KeyJ
- ike.Ctrl = true
- case 0x0B:
- ike.Code = KeyK
- ike.Ctrl = true
- case 0x0C:
- ike.Code = KeyL
- ike.Ctrl = true
- case 0x0D:
- ike.Code = KeyM
- ike.Ctrl = true
- case 0x0E:
- ike.Code = KeyN
- ike.Ctrl = true
- case 0x0F:
- ike.Code = KeyO
- ike.Ctrl = true
- case 0x10:
- ike.Code = KeyP
- ike.Ctrl = true
- case 0x11:
- ike.Code = KeyQ
- ike.Ctrl = true
- case 0x12:
- ike.Code = KeyR
- ike.Ctrl = true
- case 0x13:
- ike.Code = KeyS
- ike.Ctrl = true
- case 0x14:
- ike.Code = KeyT
- ike.Ctrl = true
- case 0x15:
- ike.Code = KeyU
- ike.Ctrl = true
- case 0x16:
- ike.Code = KeyV
- ike.Ctrl = true
- case 0x17:
- ike.Code = KeyW
- ike.Ctrl = true
- case 0x18:
- ike.Code = KeyX
- ike.Ctrl = true
- case 0x19:
- ike.Code = KeyY
- ike.Ctrl = true
- case 0x1a:
- ike.Code = KeyZ
- ike.Ctrl = true
- default:
- ike.Code = KeyUnknown
- }
-
-
- switch de {
- case '\t':
- ike.Code = KeyTab
- ike.Rune = 0x00
- case '\r':
- ike.Code = KeyEnter
- ike.Rune = 0x00
- case 127:
- ike.Code = KeyBackspace
- ike.Rune = 0x00
- case 27:
- ike.Code = KeyEsc
- ike.Rune = 0x00
- }
- }
- }
- return ike
- }
- type termKey uint16
- const (
- termKeyF1 termKey = 0xFFFF - iota
- termKeyF2
- termKeyF3
- termKeyF4
- termKeyF5
- termKeyF6
- termKeyF7
- termKeyF8
- termKeyF9
- termKeyF10
- termKeyF11
- termKeyF12
- termKeyInsert
- termKeyDelete
- termKeyHome
- termKeyHome1
- termKeyEnd
- termKeyEnd1
- termKeyPgup
- termKeyPgdn
- termKeyArrowUp
- termKeyArrowUp1
- termKeyArrowDown
- termKeyArrowDown1
- termKeyArrowLeft
- termKeyArrowLeft1
- termKeyArrowRight
- termKeyArrowRight1
- )
- var (
- etermKeys = []string{
- "\x1b[11~", "\x1b[12~", "\x1b[13~", "\x1b[14~", "\x1b[15~",
- "\x1b[17~", "\x1b[18~", "\x1b[19~", "\x1b[20~", "\x1b[21~",
- "\x1b[23~", "\x1b[24~", "\x1b[2~", "\x1b[3~", "\x1b[7~",
- "\x1b[7~", "\x1b[8~", "\x1b[8~", "\x1b[5~", "\x1b[6~",
- "\x1b[A", "\x1bOA", "\x1b[B", "\x1bOB", "\x1b[D", "\x1bOD",
- "\x1b[C", "\x1bOC",
- }
- screenKeys = []string{
- "\x1bOP", "\x1bOQ", "\x1bOR", "\x1bOS", "\x1b[15~", "\x1b[17~",
- "\x1b[18~", "\x1b[19~", "\x1b[20~", "\x1b[21~", "\x1b[23~",
- "\x1b[24~", "\x1b[2~", "\x1b[3~", "\x1b[1~", "\x1b[1~", "\x1b[4~",
- "\x1b[4~", "\x1b[5~", "\x1b[6~", "\x1b[A", "\x1bOA", "\x1b[B",
- "\x1bOB", "\x1b[D", "\x1bOD", "\x1b[C", "\x1bOC",
- }
- xtermKeys = []string{
- "\x1bOP", "\x1bOQ", "\x1bOR", "\x1bOS", "\x1b[15~", "\x1b[17~",
- "\x1b[18~", "\x1b[19~", "\x1b[20~", "\x1b[21~", "\x1b[23~",
- "\x1b[24~", "\x1b[2~", "\x1b[3~", "\x1b[H", "\x1bOH", "\x1b[F",
- "\x1bOF", "\x1b[5~", "\x1b[6~", "\x1b[A", "\x1bOA", "\x1b[B",
- "\x1bOB", "\x1b[D", "\x1bOD", "\x1b[C", "\x1bOC",
- }
- rxvtKeys = []string{
- "\x1b[11~", "\x1b[12~", "\x1b[13~", "\x1b[14~", "\x1b[15~",
- "\x1b[17~", "\x1b[18~", "\x1b[19~", "\x1b[20~", "\x1b[21~",
- "\x1b[23~", "\x1b[24~", "\x1b[2~", "\x1b[3~", "\x1b[7~",
- "\x1b[7~", "\x1b[8~", "\x1b[8~", "\x1b[5~", "\x1b[6~",
- "\x1b[A", "\x1bOA", "\x1b[B", "\x1bOB", "\x1b[D", "\x1bOD",
- "\x1b[C", "\x1bOC",
- }
- linuxKeys = []string{
- "\x1b[[A", "\x1b[[B", "\x1b[[C", "\x1b[[D", "\x1b[[E",
- "\x1b[17~", "\x1b[18~", "\x1b[19~", "\x1b[20~", "\x1b[21~",
- "\x1b[23~", "\x1b[24~", "\x1b[2~", "\x1b[3~", "\x1b[1~",
- "\x1b[1~", "\x1b[4~", "\x1b[4~", "\x1b[5~", "\x1b[6~",
- "\x1b[A", "\x1bOA", "\x1b[B", "\x1bOB", "\x1b[D", "\x1bOD",
- "\x1b[C", "\x1bOC",
- }
- DefaultTermMappings = []struct {
- name string
- keys []string
- }{
- {"Eterm", etermKeys},
- {"screen", screenKeys},
- {"xterm", xtermKeys},
- {"xterm-256color", xtermKeys},
- {"rxvt-unicode", rxvtKeys},
- {"rxvt-256color", rxvtKeys},
- {"linux", linuxKeys},
- }
- CompatibilityTermMappings = []struct {
- namePart string
- keys []string
- }{
- {"xterm", xtermKeys},
- {"rxvt", rxvtKeys},
- {"linux", linuxKeys},
- {"Eterm", etermKeys},
- {"screen", screenKeys},
- {"cygwin", xtermKeys},
- {"st", xtermKeys},
- }
- )
- func detectTerminal() ([]string, error) {
- var keys []string
- name := os.Getenv("TERM")
- if name == "" {
- return nil, fmt.Errorf("Cannot determine terminal - TERM environment variable not set")
- }
- err := fmt.Errorf("Terminal %s is not supported", name)
-
- for _, t := range DefaultTermMappings {
- if name == t.name {
- keys = t.keys
- err = nil
- break
- }
- }
- if keys == nil {
-
- for _, t := range CompatibilityTermMappings {
- if strings.Contains(name, t.namePart) {
- keys = t.keys
- err = nil
- break
- }
- }
- }
- return keys, err
- }
- func fcntl(fd uintptr, cmd int, arg int) error {
- var err error
- _, _, e := syscall.Syscall(syscall.SYS_FCNTL, fd, uintptr(cmd),
- uintptr(arg))
-
- if e != 0 {
- err = os.NewSyscallError("SYS_FCNTL", e)
- }
- return err
- }
- func ioctl(fd uintptr, cmd int, termios *syscall.Termios) error {
- var err error
- r, _, e := syscall.Syscall(syscall.SYS_IOCTL, fd, uintptr(cmd),
- uintptr(unsafe.Pointer(termios)))
- if r != 0 {
- err = os.NewSyscallError("SYS_IOCTL", e)
- }
- return err
- }
|