2017-01-07 16:30:31 +00:00
|
|
|
package tui
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"os"
|
2017-01-11 12:48:36 +00:00
|
|
|
"os/exec"
|
2017-01-23 03:47:55 +00:00
|
|
|
"regexp"
|
2017-01-07 16:30:31 +00:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
"syscall"
|
|
|
|
"time"
|
|
|
|
"unicode/utf8"
|
|
|
|
|
|
|
|
"github.com/junegunn/fzf/src/util"
|
2017-01-15 04:10:59 +00:00
|
|
|
|
|
|
|
"golang.org/x/crypto/ssh/terminal"
|
2017-01-07 16:30:31 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
|
|
|
defaultWidth = 80
|
|
|
|
defaultHeight = 24
|
|
|
|
|
2017-01-23 03:51:31 +00:00
|
|
|
defaultEscDelay = 100
|
2017-01-07 16:30:31 +00:00
|
|
|
escPollInterval = 5
|
2017-01-23 03:47:55 +00:00
|
|
|
offsetPollTries = 10
|
2017-01-07 16:30:31 +00:00
|
|
|
)
|
|
|
|
|
2017-01-15 04:10:59 +00:00
|
|
|
const consoleDevice string = "/dev/tty"
|
|
|
|
|
2017-01-23 03:47:55 +00:00
|
|
|
var offsetRegexp *regexp.Regexp = regexp.MustCompile("\x1b\\[([0-9]+);([0-9]+)R")
|
|
|
|
|
2017-01-07 16:30:31 +00:00
|
|
|
func openTtyIn() *os.File {
|
2017-01-15 04:10:59 +00:00
|
|
|
in, err := os.OpenFile(consoleDevice, syscall.O_RDONLY, 0)
|
2017-01-07 16:30:31 +00:00
|
|
|
if err != nil {
|
2017-01-15 04:10:59 +00:00
|
|
|
panic("Failed to open " + consoleDevice)
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
return in
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) stderr(str string) {
|
2017-01-15 17:26:36 +00:00
|
|
|
r.stderrInternal(str, true)
|
|
|
|
}
|
|
|
|
|
|
|
|
// FIXME: Need better handling of non-displayable characters
|
|
|
|
func (r *LightRenderer) stderrInternal(str string, allowNLCR bool) {
|
2017-01-07 16:30:31 +00:00
|
|
|
bytes := []byte(str)
|
|
|
|
runes := []rune{}
|
|
|
|
for len(bytes) > 0 {
|
|
|
|
r, sz := utf8.DecodeRune(bytes)
|
2017-01-15 17:26:36 +00:00
|
|
|
if r == utf8.RuneError || r < 32 &&
|
|
|
|
r != '\x1b' && (!allowNLCR || r != '\n' && r != '\r') {
|
2017-01-07 16:30:31 +00:00
|
|
|
runes = append(runes, '?')
|
|
|
|
} else {
|
|
|
|
runes = append(runes, r)
|
|
|
|
}
|
|
|
|
bytes = bytes[sz:]
|
|
|
|
}
|
|
|
|
r.queued += string(runes)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) csi(code string) {
|
|
|
|
r.stderr("\x1b[" + code)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) flush() {
|
|
|
|
if len(r.queued) > 0 {
|
|
|
|
fmt.Fprint(os.Stderr, r.queued)
|
|
|
|
r.queued = ""
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Light renderer
|
|
|
|
type LightRenderer struct {
|
|
|
|
theme *ColorTheme
|
|
|
|
mouse bool
|
|
|
|
forceBlack bool
|
2017-03-04 02:29:31 +00:00
|
|
|
clearOnExit bool
|
2017-01-07 16:30:31 +00:00
|
|
|
prevDownTime time.Time
|
|
|
|
clickY []int
|
|
|
|
ttyin *os.File
|
|
|
|
buffer []byte
|
2017-01-15 04:10:59 +00:00
|
|
|
origState *terminal.State
|
2017-01-07 16:30:31 +00:00
|
|
|
width int
|
|
|
|
height int
|
|
|
|
yoffset int
|
|
|
|
tabstop int
|
|
|
|
escDelay int
|
2017-01-21 20:26:38 +00:00
|
|
|
fullscreen bool
|
2017-01-07 16:30:31 +00:00
|
|
|
upOneLine bool
|
|
|
|
queued string
|
2017-01-09 10:09:30 +00:00
|
|
|
y int
|
|
|
|
x int
|
2017-01-07 16:30:31 +00:00
|
|
|
maxHeightFunc func(int) int
|
|
|
|
}
|
|
|
|
|
|
|
|
type LightWindow struct {
|
|
|
|
renderer *LightRenderer
|
|
|
|
colored bool
|
2017-02-04 12:51:22 +00:00
|
|
|
border BorderStyle
|
2017-01-07 16:30:31 +00:00
|
|
|
top int
|
|
|
|
left int
|
|
|
|
width int
|
|
|
|
height int
|
|
|
|
posx int
|
|
|
|
posy int
|
|
|
|
tabstop int
|
|
|
|
bg Color
|
|
|
|
}
|
|
|
|
|
2017-03-04 05:09:36 +00:00
|
|
|
func NewLightRenderer(theme *ColorTheme, forceBlack bool, mouse bool, tabstop int, clearOnExit bool, fullscreen bool, maxHeightFunc func(int) int) Renderer {
|
2017-01-07 16:30:31 +00:00
|
|
|
r := LightRenderer{
|
|
|
|
theme: theme,
|
|
|
|
forceBlack: forceBlack,
|
|
|
|
mouse: mouse,
|
2017-03-04 02:29:31 +00:00
|
|
|
clearOnExit: clearOnExit,
|
2017-01-07 16:30:31 +00:00
|
|
|
ttyin: openTtyIn(),
|
2017-01-21 20:26:38 +00:00
|
|
|
yoffset: 0,
|
2017-01-07 16:30:31 +00:00
|
|
|
tabstop: tabstop,
|
2017-03-04 05:09:36 +00:00
|
|
|
fullscreen: fullscreen,
|
2017-01-07 16:30:31 +00:00
|
|
|
upOneLine: false,
|
|
|
|
maxHeightFunc: maxHeightFunc}
|
|
|
|
return &r
|
|
|
|
}
|
|
|
|
|
2017-01-15 04:10:59 +00:00
|
|
|
func (r *LightRenderer) fd() int {
|
|
|
|
return int(r.ttyin.Fd())
|
|
|
|
}
|
|
|
|
|
2017-01-07 16:30:31 +00:00
|
|
|
func (r *LightRenderer) defaultTheme() *ColorTheme {
|
2017-01-15 04:10:59 +00:00
|
|
|
if strings.Contains(os.Getenv("TERM"), "256") {
|
|
|
|
return Dark256
|
|
|
|
}
|
2017-01-11 12:48:36 +00:00
|
|
|
colors, err := exec.Command("tput", "colors").Output()
|
2017-01-07 16:30:31 +00:00
|
|
|
if err == nil && atoi(strings.TrimSpace(string(colors)), 16) > 16 {
|
|
|
|
return Dark256
|
|
|
|
}
|
|
|
|
return Default16
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) findOffset() (row int, col int) {
|
|
|
|
r.csi("6n")
|
|
|
|
r.flush()
|
2017-01-23 03:47:55 +00:00
|
|
|
bytes := []byte{}
|
|
|
|
for tries := 0; tries < offsetPollTries; tries++ {
|
|
|
|
bytes = r.getBytesInternal(bytes, tries > 0)
|
|
|
|
offsets := offsetRegexp.FindSubmatch(bytes)
|
|
|
|
if len(offsets) > 2 {
|
|
|
|
return atoi(string(offsets[1]), 0) - 1, atoi(string(offsets[2]), 0) - 1
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return -1, -1
|
|
|
|
}
|
|
|
|
|
|
|
|
func repeat(s string, times int) string {
|
|
|
|
if times > 0 {
|
|
|
|
return strings.Repeat(s, times)
|
|
|
|
}
|
|
|
|
return ""
|
|
|
|
}
|
|
|
|
|
|
|
|
func atoi(s string, defaultValue int) int {
|
|
|
|
value, err := strconv.Atoi(s)
|
|
|
|
if err != nil {
|
|
|
|
return defaultValue
|
|
|
|
}
|
|
|
|
return value
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) Init() {
|
2017-01-23 03:51:31 +00:00
|
|
|
r.escDelay = atoi(os.Getenv("ESCDELAY"), defaultEscDelay)
|
2017-01-07 16:30:31 +00:00
|
|
|
|
2017-01-15 04:10:59 +00:00
|
|
|
fd := r.fd()
|
|
|
|
origState, err := terminal.GetState(fd)
|
|
|
|
if err != nil {
|
|
|
|
errorExit(err.Error())
|
|
|
|
}
|
|
|
|
r.origState = origState
|
|
|
|
terminal.MakeRaw(fd)
|
2017-03-04 05:09:36 +00:00
|
|
|
r.updateTerminalSize()
|
2017-01-07 16:30:31 +00:00
|
|
|
initTheme(r.theme, r.defaultTheme(), r.forceBlack)
|
|
|
|
|
2017-01-21 20:26:38 +00:00
|
|
|
if r.fullscreen {
|
|
|
|
r.smcup()
|
|
|
|
} else {
|
2017-07-18 11:50:38 +00:00
|
|
|
// We assume that --no-clear is used for repetitive relaunching of fzf.
|
|
|
|
// So we do not clear the lower bottom of the screen.
|
|
|
|
if r.clearOnExit {
|
|
|
|
r.csi("J")
|
|
|
|
}
|
2017-01-23 03:15:31 +00:00
|
|
|
y, x := r.findOffset()
|
|
|
|
r.mouse = r.mouse && y >= 0
|
2017-07-18 11:50:38 +00:00
|
|
|
// When --no-clear is used for repetitive relaunching, there is a small
|
|
|
|
// time frame between fzf processes where the user keystrokes are not
|
|
|
|
// captured by either of fzf process which can cause x offset to be
|
|
|
|
// increased and we're left with unwanted extra new line.
|
|
|
|
if x > 0 && r.clearOnExit {
|
2017-01-21 20:26:38 +00:00
|
|
|
r.upOneLine = true
|
|
|
|
r.makeSpace()
|
|
|
|
}
|
|
|
|
for i := 1; i < r.MaxY(); i++ {
|
|
|
|
r.makeSpace()
|
|
|
|
}
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if r.mouse {
|
|
|
|
r.csi("?1000h")
|
|
|
|
}
|
|
|
|
r.csi(fmt.Sprintf("%dA", r.MaxY()-1))
|
|
|
|
r.csi("G")
|
2017-01-21 18:19:50 +00:00
|
|
|
r.csi("K")
|
2017-07-18 11:50:38 +00:00
|
|
|
r.csi("s")
|
2017-01-21 20:26:38 +00:00
|
|
|
if !r.fullscreen && r.mouse {
|
2017-01-10 13:55:55 +00:00
|
|
|
r.yoffset, _ = r.findOffset()
|
|
|
|
}
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
|
2017-01-21 18:19:50 +00:00
|
|
|
func (r *LightRenderer) makeSpace() {
|
|
|
|
r.stderr("\n")
|
|
|
|
r.csi("G")
|
|
|
|
}
|
|
|
|
|
2017-01-09 10:09:30 +00:00
|
|
|
func (r *LightRenderer) move(y int, x int) {
|
|
|
|
// w.csi("u")
|
|
|
|
if r.y < y {
|
|
|
|
r.csi(fmt.Sprintf("%dB", y-r.y))
|
|
|
|
} else if r.y > y {
|
|
|
|
r.csi(fmt.Sprintf("%dA", r.y-y))
|
|
|
|
}
|
|
|
|
r.stderr("\r")
|
|
|
|
if x > 0 {
|
|
|
|
r.csi(fmt.Sprintf("%dC", x))
|
|
|
|
}
|
|
|
|
r.y = y
|
|
|
|
r.x = x
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) origin() {
|
|
|
|
r.move(0, 0)
|
|
|
|
}
|
|
|
|
|
2017-01-15 04:10:59 +00:00
|
|
|
func getEnv(name string, defaultValue int) int {
|
|
|
|
env := os.Getenv(name)
|
|
|
|
if len(env) == 0 {
|
|
|
|
return defaultValue
|
|
|
|
}
|
|
|
|
return atoi(env, defaultValue)
|
|
|
|
}
|
|
|
|
|
2017-03-04 05:09:36 +00:00
|
|
|
func (r *LightRenderer) updateTerminalSize() {
|
2017-01-15 04:10:59 +00:00
|
|
|
width, height, err := terminal.GetSize(r.fd())
|
|
|
|
if err == nil {
|
|
|
|
r.width = width
|
2017-03-04 05:09:36 +00:00
|
|
|
r.height = r.maxHeightFunc(height)
|
2017-01-07 16:30:31 +00:00
|
|
|
} else {
|
2017-01-15 04:10:59 +00:00
|
|
|
r.width = getEnv("COLUMNS", defaultWidth)
|
|
|
|
r.height = r.maxHeightFunc(getEnv("LINES", defaultHeight))
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-10 17:12:32 +00:00
|
|
|
func (r *LightRenderer) getch(nonblock bool) (int, bool) {
|
2017-01-07 16:30:31 +00:00
|
|
|
b := make([]byte, 1)
|
2017-05-24 16:36:59 +00:00
|
|
|
fd := r.fd()
|
2017-01-07 16:30:31 +00:00
|
|
|
util.SetNonblock(r.ttyin, nonblock)
|
2017-05-24 16:36:59 +00:00
|
|
|
_, err := util.Read(fd, b)
|
2017-01-07 16:30:31 +00:00
|
|
|
if err != nil {
|
2017-01-10 17:12:32 +00:00
|
|
|
return 0, false
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
2017-01-10 17:12:32 +00:00
|
|
|
return int(b[0]), true
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) getBytes() []byte {
|
2017-01-23 03:47:55 +00:00
|
|
|
return r.getBytesInternal(r.buffer, false)
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
|
2017-01-23 03:47:55 +00:00
|
|
|
func (r *LightRenderer) getBytesInternal(buffer []byte, nonblock bool) []byte {
|
|
|
|
c, ok := r.getch(nonblock)
|
|
|
|
if !nonblock && !ok {
|
2017-01-10 17:12:32 +00:00
|
|
|
r.Close()
|
2017-01-15 04:10:59 +00:00
|
|
|
errorExit("Failed to read " + consoleDevice)
|
2017-01-10 17:12:32 +00:00
|
|
|
}
|
2017-01-07 16:30:31 +00:00
|
|
|
|
|
|
|
retries := 0
|
2017-01-23 03:47:55 +00:00
|
|
|
if c == ESC || nonblock {
|
2017-01-07 16:30:31 +00:00
|
|
|
retries = r.escDelay / escPollInterval
|
|
|
|
}
|
|
|
|
buffer = append(buffer, byte(c))
|
|
|
|
|
|
|
|
for {
|
2017-01-10 17:12:32 +00:00
|
|
|
c, ok = r.getch(true)
|
|
|
|
if !ok {
|
2017-01-07 16:30:31 +00:00
|
|
|
if retries > 0 {
|
|
|
|
retries--
|
|
|
|
time.Sleep(escPollInterval * time.Millisecond)
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
break
|
|
|
|
}
|
|
|
|
retries = 0
|
|
|
|
buffer = append(buffer, byte(c))
|
|
|
|
}
|
|
|
|
|
|
|
|
return buffer
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) GetChar() Event {
|
|
|
|
if len(r.buffer) == 0 {
|
|
|
|
r.buffer = r.getBytes()
|
|
|
|
}
|
|
|
|
if len(r.buffer) == 0 {
|
|
|
|
panic("Empty buffer")
|
|
|
|
}
|
|
|
|
|
|
|
|
sz := 1
|
|
|
|
defer func() {
|
|
|
|
r.buffer = r.buffer[sz:]
|
|
|
|
}()
|
|
|
|
|
|
|
|
switch r.buffer[0] {
|
|
|
|
case CtrlC:
|
|
|
|
return Event{CtrlC, 0, nil}
|
|
|
|
case CtrlG:
|
|
|
|
return Event{CtrlG, 0, nil}
|
|
|
|
case CtrlQ:
|
|
|
|
return Event{CtrlQ, 0, nil}
|
|
|
|
case 127:
|
|
|
|
return Event{BSpace, 0, nil}
|
2017-01-27 17:54:47 +00:00
|
|
|
case 0:
|
|
|
|
return Event{CtrlSpace, 0, nil}
|
2017-01-07 16:30:31 +00:00
|
|
|
case ESC:
|
|
|
|
ev := r.escSequence(&sz)
|
|
|
|
// Second chance
|
|
|
|
if ev.Type == Invalid {
|
|
|
|
r.buffer = r.getBytes()
|
|
|
|
ev = r.escSequence(&sz)
|
|
|
|
}
|
|
|
|
return ev
|
|
|
|
}
|
|
|
|
|
|
|
|
// CTRL-A ~ CTRL-Z
|
|
|
|
if r.buffer[0] <= CtrlZ {
|
|
|
|
return Event{int(r.buffer[0]), 0, nil}
|
|
|
|
}
|
|
|
|
char, rsz := utf8.DecodeRune(r.buffer)
|
|
|
|
if char == utf8.RuneError {
|
|
|
|
return Event{ESC, 0, nil}
|
|
|
|
}
|
|
|
|
sz = rsz
|
|
|
|
return Event{Rune, char, nil}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) escSequence(sz *int) Event {
|
|
|
|
if len(r.buffer) < 2 {
|
|
|
|
return Event{ESC, 0, nil}
|
|
|
|
}
|
|
|
|
*sz = 2
|
2017-04-27 17:36:36 +00:00
|
|
|
if r.buffer[1] >= 1 && r.buffer[1] <= 'z'-'a'+1 {
|
|
|
|
return Event{int(CtrlAltA + r.buffer[1] - 1), 0, nil}
|
|
|
|
}
|
2017-01-07 16:30:31 +00:00
|
|
|
switch r.buffer[1] {
|
|
|
|
case 32:
|
|
|
|
return Event{AltSpace, 0, nil}
|
|
|
|
case 47:
|
|
|
|
return Event{AltSlash, 0, nil}
|
|
|
|
case 98:
|
|
|
|
return Event{AltB, 0, nil}
|
|
|
|
case 100:
|
|
|
|
return Event{AltD, 0, nil}
|
|
|
|
case 102:
|
|
|
|
return Event{AltF, 0, nil}
|
|
|
|
case 127:
|
|
|
|
return Event{AltBS, 0, nil}
|
|
|
|
case 91, 79:
|
|
|
|
if len(r.buffer) < 3 {
|
|
|
|
return Event{Invalid, 0, nil}
|
|
|
|
}
|
|
|
|
*sz = 3
|
|
|
|
switch r.buffer[2] {
|
|
|
|
case 68:
|
|
|
|
return Event{Left, 0, nil}
|
|
|
|
case 67:
|
|
|
|
return Event{Right, 0, nil}
|
|
|
|
case 66:
|
|
|
|
return Event{Down, 0, nil}
|
|
|
|
case 65:
|
|
|
|
return Event{Up, 0, nil}
|
|
|
|
case 90:
|
|
|
|
return Event{BTab, 0, nil}
|
|
|
|
case 72:
|
|
|
|
return Event{Home, 0, nil}
|
|
|
|
case 70:
|
|
|
|
return Event{End, 0, nil}
|
|
|
|
case 77:
|
|
|
|
return r.mouseSequence(sz)
|
|
|
|
case 80:
|
|
|
|
return Event{F1, 0, nil}
|
|
|
|
case 81:
|
|
|
|
return Event{F2, 0, nil}
|
|
|
|
case 82:
|
|
|
|
return Event{F3, 0, nil}
|
|
|
|
case 83:
|
|
|
|
return Event{F4, 0, nil}
|
|
|
|
case 49, 50, 51, 52, 53, 54:
|
|
|
|
if len(r.buffer) < 4 {
|
|
|
|
return Event{Invalid, 0, nil}
|
|
|
|
}
|
|
|
|
*sz = 4
|
|
|
|
switch r.buffer[2] {
|
|
|
|
case 50:
|
|
|
|
if len(r.buffer) == 5 && r.buffer[4] == 126 {
|
|
|
|
*sz = 5
|
|
|
|
switch r.buffer[3] {
|
|
|
|
case 48:
|
|
|
|
return Event{F9, 0, nil}
|
|
|
|
case 49:
|
|
|
|
return Event{F10, 0, nil}
|
|
|
|
case 51:
|
|
|
|
return Event{F11, 0, nil}
|
|
|
|
case 52:
|
|
|
|
return Event{F12, 0, nil}
|
|
|
|
}
|
|
|
|
}
|
2017-06-21 17:35:57 +00:00
|
|
|
// Bracketed paste mode: \e[200~ ... \e[201~
|
|
|
|
if r.buffer[3] == '0' && (r.buffer[4] == '0' || r.buffer[4] == '1') && r.buffer[5] == '~' {
|
|
|
|
// Immediately discard the sequence from the buffer and reread input
|
|
|
|
r.buffer = r.buffer[6:]
|
|
|
|
*sz = 0
|
|
|
|
return r.GetChar()
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
return Event{Invalid, 0, nil} // INS
|
|
|
|
case 51:
|
|
|
|
return Event{Del, 0, nil}
|
|
|
|
case 52:
|
|
|
|
return Event{End, 0, nil}
|
|
|
|
case 53:
|
|
|
|
return Event{PgUp, 0, nil}
|
|
|
|
case 54:
|
|
|
|
return Event{PgDn, 0, nil}
|
|
|
|
case 49:
|
|
|
|
switch r.buffer[3] {
|
|
|
|
case 126:
|
|
|
|
return Event{Home, 0, nil}
|
|
|
|
case 53, 55, 56, 57:
|
|
|
|
if len(r.buffer) == 5 && r.buffer[4] == 126 {
|
|
|
|
*sz = 5
|
|
|
|
switch r.buffer[3] {
|
|
|
|
case 53:
|
|
|
|
return Event{F5, 0, nil}
|
|
|
|
case 55:
|
|
|
|
return Event{F6, 0, nil}
|
|
|
|
case 56:
|
|
|
|
return Event{F7, 0, nil}
|
|
|
|
case 57:
|
|
|
|
return Event{F8, 0, nil}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return Event{Invalid, 0, nil}
|
|
|
|
case 59:
|
|
|
|
if len(r.buffer) != 6 {
|
|
|
|
return Event{Invalid, 0, nil}
|
|
|
|
}
|
|
|
|
*sz = 6
|
|
|
|
switch r.buffer[4] {
|
|
|
|
case 50:
|
|
|
|
switch r.buffer[5] {
|
|
|
|
case 68:
|
|
|
|
return Event{Home, 0, nil}
|
|
|
|
case 67:
|
|
|
|
return Event{End, 0, nil}
|
|
|
|
}
|
|
|
|
case 53:
|
|
|
|
switch r.buffer[5] {
|
|
|
|
case 68:
|
|
|
|
return Event{SLeft, 0, nil}
|
|
|
|
case 67:
|
|
|
|
return Event{SRight, 0, nil}
|
|
|
|
}
|
|
|
|
} // r.buffer[4]
|
|
|
|
} // r.buffer[3]
|
|
|
|
} // r.buffer[2]
|
|
|
|
} // r.buffer[2]
|
|
|
|
} // r.buffer[1]
|
|
|
|
if r.buffer[1] >= 'a' && r.buffer[1] <= 'z' {
|
|
|
|
return Event{AltA + int(r.buffer[1]) - 'a', 0, nil}
|
|
|
|
}
|
|
|
|
return Event{Invalid, 0, nil}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) mouseSequence(sz *int) Event {
|
2017-01-21 20:26:38 +00:00
|
|
|
if len(r.buffer) < 6 || !r.mouse {
|
2017-01-07 16:30:31 +00:00
|
|
|
return Event{Invalid, 0, nil}
|
|
|
|
}
|
|
|
|
*sz = 6
|
|
|
|
switch r.buffer[3] {
|
|
|
|
case 32, 36, 40, 48, // mouse-down / shift / cmd / ctrl
|
|
|
|
35, 39, 43, 51: // mouse-up / shift / cmd / ctrl
|
|
|
|
mod := r.buffer[3] >= 36
|
|
|
|
down := r.buffer[3]%2 == 0
|
|
|
|
x := int(r.buffer[4] - 33)
|
|
|
|
y := int(r.buffer[5]-33) - r.yoffset
|
|
|
|
double := false
|
|
|
|
if down {
|
|
|
|
now := time.Now()
|
|
|
|
if now.Sub(r.prevDownTime) < doubleClickDuration {
|
|
|
|
r.clickY = append(r.clickY, y)
|
|
|
|
} else {
|
|
|
|
r.clickY = []int{y}
|
|
|
|
}
|
|
|
|
r.prevDownTime = now
|
|
|
|
} else {
|
|
|
|
if len(r.clickY) > 1 && r.clickY[0] == r.clickY[1] &&
|
|
|
|
time.Now().Sub(r.prevDownTime) < doubleClickDuration {
|
|
|
|
double = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Event{Mouse, 0, &MouseEvent{y, x, 0, down, double, mod}}
|
|
|
|
case 96, 100, 104, 112, // scroll-up / shift / cmd / ctrl
|
|
|
|
97, 101, 105, 113: // scroll-down / shift / cmd / ctrl
|
|
|
|
mod := r.buffer[3] >= 100
|
|
|
|
s := 1 - int(r.buffer[3]%2)*2
|
|
|
|
x := int(r.buffer[4] - 33)
|
|
|
|
y := int(r.buffer[5]-33) - r.yoffset
|
|
|
|
return Event{Mouse, 0, &MouseEvent{y, x, s, false, false, mod}}
|
|
|
|
}
|
|
|
|
return Event{Invalid, 0, nil}
|
|
|
|
}
|
|
|
|
|
2017-01-21 20:26:38 +00:00
|
|
|
func (r *LightRenderer) smcup() {
|
|
|
|
r.csi("?1049h")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) rmcup() {
|
|
|
|
r.csi("?1049l")
|
|
|
|
}
|
|
|
|
|
2017-04-28 13:58:08 +00:00
|
|
|
func (r *LightRenderer) Pause(clear bool) {
|
2017-01-15 04:10:59 +00:00
|
|
|
terminal.Restore(r.fd(), r.origState)
|
2017-04-28 13:58:08 +00:00
|
|
|
if clear {
|
|
|
|
if r.fullscreen {
|
|
|
|
r.rmcup()
|
|
|
|
} else {
|
|
|
|
r.smcup()
|
|
|
|
r.csi("H")
|
|
|
|
}
|
|
|
|
r.flush()
|
2017-01-21 20:26:38 +00:00
|
|
|
}
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
|
2017-04-28 13:58:08 +00:00
|
|
|
func (r *LightRenderer) Resume(clear bool) {
|
2017-01-15 04:10:59 +00:00
|
|
|
terminal.MakeRaw(r.fd())
|
2017-04-28 13:58:08 +00:00
|
|
|
if clear {
|
|
|
|
if r.fullscreen {
|
|
|
|
r.smcup()
|
|
|
|
} else {
|
|
|
|
r.rmcup()
|
|
|
|
}
|
|
|
|
r.flush()
|
|
|
|
} else if !r.fullscreen && r.mouse {
|
|
|
|
// NOTE: Resume(false) is only called on SIGCONT after SIGSTOP.
|
|
|
|
// And It's highly likely that the offset we obtained at the beginning will
|
|
|
|
// no longer be correct, so we simply disable mouse input.
|
|
|
|
r.csi("?1000l")
|
|
|
|
r.mouse = false
|
2017-01-21 20:26:38 +00:00
|
|
|
}
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) Clear() {
|
2017-02-14 13:30:09 +00:00
|
|
|
if r.fullscreen {
|
|
|
|
r.csi("H")
|
|
|
|
}
|
2017-01-09 10:09:30 +00:00
|
|
|
// r.csi("u")
|
|
|
|
r.origin()
|
2017-01-07 16:30:31 +00:00
|
|
|
r.csi("J")
|
|
|
|
r.flush()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) RefreshWindows(windows []Window) {
|
|
|
|
r.flush()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) Refresh() {
|
|
|
|
r.updateTerminalSize()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) Close() {
|
2017-01-09 10:09:30 +00:00
|
|
|
// r.csi("u")
|
2017-03-04 02:29:31 +00:00
|
|
|
if r.clearOnExit {
|
|
|
|
if r.fullscreen {
|
|
|
|
r.rmcup()
|
|
|
|
} else {
|
|
|
|
r.origin()
|
|
|
|
if r.upOneLine {
|
|
|
|
r.csi("A")
|
|
|
|
}
|
|
|
|
r.csi("J")
|
2017-01-21 20:26:38 +00:00
|
|
|
}
|
2017-07-18 11:50:38 +00:00
|
|
|
} else if !r.fullscreen {
|
|
|
|
r.csi("u")
|
2017-01-21 20:26:38 +00:00
|
|
|
}
|
2017-01-07 16:30:31 +00:00
|
|
|
if r.mouse {
|
|
|
|
r.csi("?1000l")
|
|
|
|
}
|
|
|
|
r.flush()
|
2017-01-15 04:10:59 +00:00
|
|
|
terminal.Restore(r.fd(), r.origState)
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) MaxX() int {
|
|
|
|
return r.width
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) MaxY() int {
|
|
|
|
return r.height
|
|
|
|
}
|
|
|
|
|
|
|
|
func (r *LightRenderer) DoesAutoWrap() bool {
|
2017-02-18 14:49:00 +00:00
|
|
|
return false
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
|
2017-01-15 17:26:36 +00:00
|
|
|
func (r *LightRenderer) IsOptimized() bool {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2017-02-04 12:51:22 +00:00
|
|
|
func (r *LightRenderer) NewWindow(top int, left int, width int, height int, borderStyle BorderStyle) Window {
|
2017-01-07 16:30:31 +00:00
|
|
|
w := &LightWindow{
|
|
|
|
renderer: r,
|
|
|
|
colored: r.theme != nil,
|
2017-02-04 12:51:22 +00:00
|
|
|
border: borderStyle,
|
2017-01-07 16:30:31 +00:00
|
|
|
top: top,
|
|
|
|
left: left,
|
|
|
|
width: width,
|
|
|
|
height: height,
|
|
|
|
tabstop: r.tabstop,
|
|
|
|
bg: colDefault}
|
|
|
|
if r.theme != nil {
|
|
|
|
w.bg = r.theme.Bg
|
|
|
|
}
|
2017-02-04 12:51:22 +00:00
|
|
|
w.drawBorder()
|
2017-01-07 16:30:31 +00:00
|
|
|
return w
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) drawBorder() {
|
2017-02-04 12:51:22 +00:00
|
|
|
switch w.border {
|
|
|
|
case BorderAround:
|
|
|
|
w.drawBorderAround()
|
|
|
|
case BorderHorizontal:
|
|
|
|
w.drawBorderHorizontal()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) drawBorderHorizontal() {
|
|
|
|
w.Move(0, 0)
|
|
|
|
w.CPrint(ColBorder, AttrRegular, repeat("─", w.width))
|
|
|
|
w.Move(w.height-1, 0)
|
|
|
|
w.CPrint(ColBorder, AttrRegular, repeat("─", w.width))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) drawBorderAround() {
|
2017-01-07 16:30:31 +00:00
|
|
|
w.Move(0, 0)
|
|
|
|
w.CPrint(ColBorder, AttrRegular, "┌"+repeat("─", w.width-2)+"┐")
|
|
|
|
for y := 1; y < w.height-1; y++ {
|
|
|
|
w.Move(y, 0)
|
|
|
|
w.CPrint(ColBorder, AttrRegular, "│")
|
|
|
|
w.cprint2(colDefault, w.bg, AttrRegular, repeat(" ", w.width-2))
|
|
|
|
w.CPrint(ColBorder, AttrRegular, "│")
|
|
|
|
}
|
|
|
|
w.Move(w.height-1, 0)
|
|
|
|
w.CPrint(ColBorder, AttrRegular, "└"+repeat("─", w.width-2)+"┘")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) csi(code string) {
|
|
|
|
w.renderer.csi(code)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) stderr(str string) {
|
|
|
|
w.renderer.stderr(str)
|
|
|
|
}
|
|
|
|
|
2017-01-15 17:26:36 +00:00
|
|
|
func (w *LightWindow) stderrInternal(str string, allowNLCR bool) {
|
|
|
|
w.renderer.stderrInternal(str, allowNLCR)
|
|
|
|
}
|
|
|
|
|
2017-01-07 16:30:31 +00:00
|
|
|
func (w *LightWindow) Top() int {
|
|
|
|
return w.top
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) Left() int {
|
|
|
|
return w.left
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) Width() int {
|
|
|
|
return w.width
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) Height() int {
|
|
|
|
return w.height
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) Refresh() {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) Close() {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) X() int {
|
|
|
|
return w.posx
|
|
|
|
}
|
|
|
|
|
2017-07-19 13:46:16 +00:00
|
|
|
func (w *LightWindow) Y() int {
|
|
|
|
return w.posy
|
|
|
|
}
|
|
|
|
|
2017-01-07 16:30:31 +00:00
|
|
|
func (w *LightWindow) Enclose(y int, x int) bool {
|
|
|
|
return x >= w.left && x < (w.left+w.width) &&
|
|
|
|
y >= w.top && y < (w.top+w.height)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) Move(y int, x int) {
|
|
|
|
w.posx = x
|
|
|
|
w.posy = y
|
|
|
|
|
2017-01-09 10:09:30 +00:00
|
|
|
w.renderer.move(w.Top()+y, w.Left()+x)
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) MoveAndClear(y int, x int) {
|
|
|
|
w.Move(y, x)
|
|
|
|
// We should not delete preview window on the right
|
|
|
|
// csi("K")
|
|
|
|
w.Print(repeat(" ", w.width-x))
|
|
|
|
w.Move(y, x)
|
|
|
|
}
|
|
|
|
|
|
|
|
func attrCodes(attr Attr) []string {
|
|
|
|
codes := []string{}
|
|
|
|
if (attr & Bold) > 0 {
|
|
|
|
codes = append(codes, "1")
|
|
|
|
}
|
|
|
|
if (attr & Dim) > 0 {
|
|
|
|
codes = append(codes, "2")
|
|
|
|
}
|
|
|
|
if (attr & Italic) > 0 {
|
|
|
|
codes = append(codes, "3")
|
|
|
|
}
|
|
|
|
if (attr & Underline) > 0 {
|
|
|
|
codes = append(codes, "4")
|
|
|
|
}
|
|
|
|
if (attr & Blink) > 0 {
|
|
|
|
codes = append(codes, "5")
|
|
|
|
}
|
|
|
|
if (attr & Reverse) > 0 {
|
|
|
|
codes = append(codes, "7")
|
|
|
|
}
|
|
|
|
return codes
|
|
|
|
}
|
|
|
|
|
|
|
|
func colorCodes(fg Color, bg Color) []string {
|
|
|
|
codes := []string{}
|
|
|
|
appendCode := func(c Color, offset int) {
|
|
|
|
if c == colDefault {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
if c.is24() {
|
|
|
|
r := (c >> 16) & 0xff
|
|
|
|
g := (c >> 8) & 0xff
|
|
|
|
b := (c) & 0xff
|
|
|
|
codes = append(codes, fmt.Sprintf("%d;2;%d;%d;%d", 38+offset, r, g, b))
|
|
|
|
} else if c >= colBlack && c <= colWhite {
|
|
|
|
codes = append(codes, fmt.Sprintf("%d", int(c)+30+offset))
|
|
|
|
} else if c > colWhite && c < 16 {
|
|
|
|
codes = append(codes, fmt.Sprintf("%d", int(c)+90+offset-8))
|
|
|
|
} else if c >= 16 && c < 256 {
|
|
|
|
codes = append(codes, fmt.Sprintf("%d;5;%d", 38+offset, c))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
appendCode(fg, 0)
|
|
|
|
appendCode(bg, 10)
|
|
|
|
return codes
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) csiColor(fg Color, bg Color, attr Attr) bool {
|
|
|
|
codes := append(attrCodes(attr), colorCodes(fg, bg)...)
|
|
|
|
w.csi(";" + strings.Join(codes, ";") + "m")
|
|
|
|
return len(codes) > 0
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) Print(text string) {
|
|
|
|
w.cprint2(colDefault, w.bg, AttrRegular, text)
|
|
|
|
}
|
|
|
|
|
2017-02-02 04:14:27 +00:00
|
|
|
func cleanse(str string) string {
|
|
|
|
return strings.Replace(str, "\x1b", "?", -1)
|
|
|
|
}
|
|
|
|
|
2017-01-07 16:30:31 +00:00
|
|
|
func (w *LightWindow) CPrint(pair ColorPair, attr Attr, text string) {
|
|
|
|
if !w.colored {
|
|
|
|
w.csiColor(colDefault, colDefault, attrFor(pair, attr))
|
|
|
|
} else {
|
|
|
|
w.csiColor(pair.Fg(), pair.Bg(), attr)
|
|
|
|
}
|
2017-02-02 04:14:27 +00:00
|
|
|
w.stderrInternal(cleanse(text), false)
|
2017-01-07 16:30:31 +00:00
|
|
|
w.csi("m")
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) cprint2(fg Color, bg Color, attr Attr, text string) {
|
|
|
|
if w.csiColor(fg, bg, attr) {
|
|
|
|
defer w.csi("m")
|
|
|
|
}
|
2017-02-02 04:14:27 +00:00
|
|
|
w.stderrInternal(cleanse(text), false)
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
type wrappedLine struct {
|
|
|
|
text string
|
|
|
|
displayWidth int
|
|
|
|
}
|
|
|
|
|
|
|
|
func wrapLine(input string, prefixLength int, max int, tabstop int) []wrappedLine {
|
|
|
|
lines := []wrappedLine{}
|
|
|
|
width := 0
|
|
|
|
line := ""
|
|
|
|
for _, r := range input {
|
|
|
|
w := util.Max(util.RuneWidth(r, prefixLength+width, 8), 1)
|
|
|
|
width += w
|
|
|
|
str := string(r)
|
|
|
|
if r == '\t' {
|
|
|
|
str = repeat(" ", w)
|
|
|
|
}
|
|
|
|
if prefixLength+width <= max {
|
|
|
|
line += str
|
|
|
|
} else {
|
|
|
|
lines = append(lines, wrappedLine{string(line), width - w})
|
|
|
|
line = str
|
|
|
|
prefixLength = 0
|
|
|
|
width = util.RuneWidth(r, prefixLength, 8)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
lines = append(lines, wrappedLine{string(line), width})
|
|
|
|
return lines
|
|
|
|
}
|
|
|
|
|
2017-01-11 13:13:40 +00:00
|
|
|
func (w *LightWindow) fill(str string, onMove func()) FillReturn {
|
2017-01-07 16:30:31 +00:00
|
|
|
allLines := strings.Split(str, "\n")
|
|
|
|
for i, line := range allLines {
|
|
|
|
lines := wrapLine(line, w.posx, w.width, w.tabstop)
|
|
|
|
for j, wl := range lines {
|
2017-01-11 13:13:40 +00:00
|
|
|
if w.posx >= w.Width()-1 && wl.displayWidth == 0 {
|
|
|
|
if w.posy < w.height-1 {
|
2017-07-19 13:46:16 +00:00
|
|
|
w.Move(w.posy+1, 0)
|
2017-01-11 13:13:40 +00:00
|
|
|
}
|
|
|
|
return FillNextLine
|
|
|
|
}
|
2017-01-15 17:26:36 +00:00
|
|
|
w.stderrInternal(wl.text, false)
|
2017-01-07 16:30:31 +00:00
|
|
|
w.posx += wl.displayWidth
|
2017-07-19 13:46:16 +00:00
|
|
|
|
|
|
|
// Wrap line
|
2017-01-07 16:30:31 +00:00
|
|
|
if j < len(lines)-1 || i < len(allLines)-1 {
|
|
|
|
if w.posy+1 >= w.height {
|
2017-01-11 13:13:40 +00:00
|
|
|
return FillSuspend
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
2017-07-19 13:46:16 +00:00
|
|
|
w.MoveAndClear(w.posy, w.posx)
|
|
|
|
w.Move(w.posy+1, 0)
|
2017-01-07 16:30:31 +00:00
|
|
|
onMove()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-01-11 13:13:40 +00:00
|
|
|
return FillContinue
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) setBg() {
|
|
|
|
if w.bg != colDefault {
|
|
|
|
w.csiColor(colDefault, w.bg, AttrRegular)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-01-11 13:13:40 +00:00
|
|
|
func (w *LightWindow) Fill(text string) FillReturn {
|
2017-07-19 13:46:16 +00:00
|
|
|
w.Move(w.posy, w.posx)
|
2017-01-07 16:30:31 +00:00
|
|
|
w.setBg()
|
|
|
|
return w.fill(text, w.setBg)
|
|
|
|
}
|
|
|
|
|
2017-01-11 13:13:40 +00:00
|
|
|
func (w *LightWindow) CFill(fg Color, bg Color, attr Attr, text string) FillReturn {
|
2017-07-19 13:46:16 +00:00
|
|
|
w.Move(w.posy, w.posx)
|
2017-01-07 16:30:31 +00:00
|
|
|
if bg == colDefault {
|
|
|
|
bg = w.bg
|
|
|
|
}
|
|
|
|
if w.csiColor(fg, bg, attr) {
|
|
|
|
defer w.csi("m")
|
2017-07-21 07:43:00 +00:00
|
|
|
return w.fill(text, func() { w.csiColor(fg, bg, attr) })
|
2017-01-07 16:30:31 +00:00
|
|
|
}
|
|
|
|
return w.fill(text, w.setBg)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) FinishFill() {
|
2017-07-19 13:46:16 +00:00
|
|
|
w.MoveAndClear(w.posy, w.posx)
|
2017-01-07 16:30:31 +00:00
|
|
|
for y := w.posy + 1; y < w.height; y++ {
|
|
|
|
w.MoveAndClear(y, 0)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (w *LightWindow) Erase() {
|
2017-02-04 12:51:22 +00:00
|
|
|
w.drawBorder()
|
2017-01-07 16:30:31 +00:00
|
|
|
// We don't erase the window here to avoid flickering during scroll
|
|
|
|
w.Move(0, 0)
|
|
|
|
}
|