termui/render.go

165 lines
3.5 KiB
Go
Raw Normal View History

2017-01-14 14:07:43 +08:00
// Copyright 2017 Zack Guo <zack.y.guo@gmail.com>. All rights reserved.
2015-03-21 04:21:50 +08:00
// Use of this source code is governed by a MIT license that can
// be found in the LICENSE file.
2015-02-03 22:07:31 +08:00
package termui
2015-09-18 23:41:44 +08:00
import (
"image"
"io"
"sync"
2015-09-18 23:41:44 +08:00
"time"
"fmt"
"os"
"runtime/debug"
"bytes"
"github.com/maruel/panicparse/stack"
2015-09-18 23:41:44 +08:00
tm "github.com/nsf/termbox-go"
)
2015-02-03 22:07:31 +08:00
2015-03-25 05:16:43 +08:00
// Bufferer should be implemented by all renderable components.
2015-03-04 02:28:09 +08:00
type Bufferer interface {
Buffer() Buffer
2015-02-03 22:07:31 +08:00
}
2015-03-25 05:16:43 +08:00
// Init initializes termui library. This function should be called before any others.
// After initialization, the library must be finalized by 'Close' function.
2015-02-03 22:07:31 +08:00
func Init() error {
2015-05-10 07:29:22 +08:00
if err := tm.Init(); err != nil {
return err
}
2015-09-18 23:41:44 +08:00
sysEvtChs = make([]chan Event, 0)
go hookTermboxEvt()
2015-10-10 23:20:44 +08:00
2015-09-18 23:41:44 +08:00
renderJobs = make(chan []Bufferer)
//renderLock = new(sync.RWMutex)
2015-09-18 23:41:44 +08:00
2015-10-10 23:20:44 +08:00
Body = NewGrid()
Body.X = 0
Body.Y = 0
Body.BgColor = ThemeAttr("bg")
Body.Width = TermWidth()
2015-09-18 23:41:44 +08:00
DefaultEvtStream.Init()
DefaultEvtStream.Merge("termbox", NewSysEvtCh())
DefaultEvtStream.Merge("timer", NewTimerCh(time.Second))
2015-10-14 00:45:03 +08:00
DefaultEvtStream.Merge("custom", usrEvtCh)
DefaultEvtStream.Handle("/", DefaultHandler)
2015-10-08 02:25:59 +08:00
DefaultEvtStream.Handle("/sys/wnd/resize", func(e Event) {
w := e.Data.(EvtWnd)
Body.Width = w.Width
})
2015-09-18 23:41:44 +08:00
2015-10-08 02:25:59 +08:00
DefaultWgtMgr = NewWgtMgr()
DefaultEvtStream.Hook(DefaultWgtMgr.WgtHandlersHook())
go func() {
for bs := range renderJobs {
render(bs...)
}
}()
2015-05-10 07:29:22 +08:00
return nil
2015-02-03 22:07:31 +08:00
}
2015-03-25 05:16:43 +08:00
// Close finalizes termui library,
// should be called after successful initialization when termui's functionality isn't required anymore.
2015-02-04 09:56:49 +08:00
func Close() {
2015-02-03 22:07:31 +08:00
tm.Close()
}
var renderLock sync.Mutex
func termSync() {
renderLock.Lock()
tm.Sync()
termWidth, termHeight = tm.Size()
renderLock.Unlock()
}
2015-03-25 05:16:43 +08:00
// TermWidth returns the current terminal's width.
2015-03-20 20:24:48 +08:00
func TermWidth() int {
termSync()
return termWidth
2015-03-20 20:24:48 +08:00
}
2015-03-25 05:16:43 +08:00
// TermHeight returns the current terminal's height.
2015-03-20 20:24:48 +08:00
func TermHeight() int {
termSync()
return termHeight
2015-03-20 20:24:48 +08:00
}
2015-03-25 05:16:43 +08:00
// Render renders all Bufferer in the given order from left to right,
// right could overlap on left ones.
2015-10-10 23:20:44 +08:00
func render(bs ...Bufferer) {
defer func() {
if e := recover(); e != nil {
Close()
fmt.Fprintf(os.Stderr, "Captured a panic(value=%v) when rendering Bufferer. Exit termui and clean terminal...\nPrint stack trace:\n\n", e)
//debug.PrintStack()
gs, err := stack.ParseDump(bytes.NewReader(debug.Stack()), os.Stderr)
if err != nil {
debug.PrintStack()
os.Exit(1)
}
p := &stack.Palette{}
buckets := stack.SortBuckets(stack.Bucketize(gs, stack.AnyValue))
srcLen, pkgLen := stack.CalcLengths(buckets, false)
for _, bucket := range buckets {
io.WriteString(os.Stdout, p.BucketHeader(&bucket, false, len(buckets) > 1))
io.WriteString(os.Stdout, p.StackLines(&bucket.Signature, srcLen, pkgLen, false))
}
os.Exit(1)
}
}()
for _, b := range bs {
buf := b.Buffer()
// set cels in buf
for p, c := range buf.CellMap {
2015-05-10 07:29:22 +08:00
if p.In(buf.Area) {
tm.SetCell(p.X, p.Y, c.Ch, toTmAttr(c.Fg), toTmAttr(c.Bg))
}
2015-02-04 09:56:49 +08:00
}
2015-02-03 22:07:31 +08:00
}
renderLock.Lock()
// render
2015-02-03 22:07:31 +08:00
tm.Flush()
renderLock.Unlock()
}
func Clear() {
tm.Clear(tm.ColorDefault, toTmAttr(ThemeAttr("bg")))
}
func clearArea(r image.Rectangle, bg Attribute) {
for i := r.Min.X; i < r.Max.X; i++ {
for j := r.Min.Y; j < r.Max.Y; j++ {
tm.SetCell(i, j, ' ', tm.ColorDefault, toTmAttr(bg))
}
}
}
func ClearArea(r image.Rectangle, bg Attribute) {
clearArea(r, bg)
tm.Flush()
2015-02-03 22:07:31 +08:00
}
2015-09-18 23:41:44 +08:00
var renderJobs chan []Bufferer
2015-10-10 23:20:44 +08:00
func Render(bs ...Bufferer) {
//go func() { renderJobs <- bs }()
renderJobs <- bs
2015-09-18 23:41:44 +08:00
}