mirror of https://github.com/gdamore/tcell.git
184 lines
4.2 KiB
Go
184 lines
4.2 KiB
Go
|
//go:build ignore
|
||
|
// +build ignore
|
||
|
|
||
|
// Copyright 2023 The TCell Authors
|
||
|
//
|
||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||
|
// you may not use file except in compliance with the License.
|
||
|
// You may obtain a copy of the license at
|
||
|
//
|
||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||
|
//
|
||
|
// Unless required by applicable law or agreed to in writing, software
|
||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||
|
// See the License for the specific language governing permissions and
|
||
|
// limitations under the License.
|
||
|
//
|
||
|
// sixel displays a sixel and demonstrates how to use direct drawing
|
||
|
package main
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"fmt"
|
||
|
"image"
|
||
|
"image/png"
|
||
|
"log"
|
||
|
"math"
|
||
|
"os"
|
||
|
|
||
|
"github.com/gdamore/tcell/v2"
|
||
|
"github.com/gdamore/tcell/v2/encoding"
|
||
|
|
||
|
"github.com/mattn/go-runewidth"
|
||
|
"github.com/mattn/go-sixel"
|
||
|
)
|
||
|
|
||
|
type imageData struct {
|
||
|
width int // width in pixels
|
||
|
height int // height in pixels
|
||
|
data *bytes.Buffer
|
||
|
}
|
||
|
|
||
|
func emitStr(s tcell.Screen, x, y int, style tcell.Style, str string) {
|
||
|
for _, c := range str {
|
||
|
var comb []rune
|
||
|
w := runewidth.RuneWidth(c)
|
||
|
if w == 0 {
|
||
|
comb = []rune{c}
|
||
|
c = ' '
|
||
|
w = 1
|
||
|
}
|
||
|
s.SetContent(x, y, c, comb, style)
|
||
|
x += w
|
||
|
}
|
||
|
}
|
||
|
|
||
|
func displayHelloWorld(s tcell.Screen) {
|
||
|
w, h := s.Size()
|
||
|
s.Clear()
|
||
|
style := tcell.StyleDefault.Foreground(tcell.ColorCadetBlue.TrueColor()).Background(tcell.ColorWhite)
|
||
|
emitStr(s, w/2-7, h/2, style, "Hello, World!")
|
||
|
emitStr(s, w/2-9, h/2+1, tcell.StyleDefault, "Press ESC to exit.")
|
||
|
emitStr(s, w/2-18, h/2+2, tcell.StyleDefault, "Press Enter to toggle sixel lock.")
|
||
|
s.Show()
|
||
|
}
|
||
|
|
||
|
func displaySixel(s tcell.Screen, img *imageData, lock bool) {
|
||
|
tty, ok := s.Tty()
|
||
|
if !ok {
|
||
|
s.Fini()
|
||
|
log.Fatal("not a terminal")
|
||
|
}
|
||
|
ws, err := tty.WindowSize()
|
||
|
if err != nil {
|
||
|
s.Fini()
|
||
|
log.Fatal(err)
|
||
|
}
|
||
|
// Get the dimensions of a single cell
|
||
|
cw, ch := ws.CellDimensions()
|
||
|
if cw == 0 || ch == 0 {
|
||
|
s.Fini()
|
||
|
log.Fatal("terminal does not support sixel graphics")
|
||
|
return
|
||
|
}
|
||
|
|
||
|
// Calculate the image dimensions in cells. We round up to prevent
|
||
|
// drawing on a partially filled cell
|
||
|
sixelWidth := int(math.Ceil(float64(img.width) / float64(cw)))
|
||
|
sixelHeight := int(math.Ceil(float64(img.height) / float64(ch)))
|
||
|
|
||
|
sixelX := ws.Width/2 - (sixelWidth / 2) // Center the image horizontally
|
||
|
sixelY := ws.Height/2 - sixelHeight - 2
|
||
|
if sixelY < 0 {
|
||
|
sixelY = 0
|
||
|
}
|
||
|
// Lock the region where we will draw the sixel, this prevents tcell
|
||
|
// from drawing over this area
|
||
|
s.LockRegion(sixelX, sixelY, sixelWidth, sixelHeight, lock)
|
||
|
|
||
|
// Get the terminfo for our current terminal
|
||
|
ti, err := tcell.LookupTerminfo(os.Getenv("TERM"))
|
||
|
if err != nil {
|
||
|
s.Fini()
|
||
|
log.Fatal(err)
|
||
|
}
|
||
|
|
||
|
emitStr(s, sixelX, sixelY, tcell.StyleDefault, "This text is behind")
|
||
|
emitStr(s, sixelX, sixelY+1, tcell.StyleDefault, " the sixel")
|
||
|
|
||
|
// Move the cursor to our draw position
|
||
|
ti.TPuts(tty, ti.TGoto(sixelX, sixelY))
|
||
|
// Draw the sixel data
|
||
|
ti.TPuts(tty, img.data.String())
|
||
|
|
||
|
s.Show()
|
||
|
}
|
||
|
|
||
|
func loadImage(path string) (image.Image, error) {
|
||
|
f, err := os.Open(path)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
return png.Decode(f)
|
||
|
}
|
||
|
|
||
|
func main() {
|
||
|
encoding.Register()
|
||
|
|
||
|
s, e := tcell.NewScreen()
|
||
|
if e != nil {
|
||
|
fmt.Fprintf(os.Stderr, "%v\n", e)
|
||
|
os.Exit(1)
|
||
|
}
|
||
|
if e := s.Init(); e != nil {
|
||
|
fmt.Fprintf(os.Stderr, "%v\n", e)
|
||
|
os.Exit(1)
|
||
|
}
|
||
|
|
||
|
defStyle := tcell.StyleDefault.
|
||
|
Background(tcell.ColorBlack).
|
||
|
Foreground(tcell.ColorWhite)
|
||
|
s.SetStyle(defStyle)
|
||
|
|
||
|
raw, err := loadImage("./logos/tcell.png")
|
||
|
if err != nil {
|
||
|
s.Fini()
|
||
|
log.Println("couldn't load image. try running from the root directory")
|
||
|
log.Fatalf(" go run ./_demos/sixel.go")
|
||
|
}
|
||
|
|
||
|
img := &imageData{
|
||
|
width: raw.Bounds().Dx(),
|
||
|
height: raw.Bounds().Dy(),
|
||
|
data: bytes.NewBuffer(nil),
|
||
|
}
|
||
|
enc := sixel.NewEncoder(img.data)
|
||
|
if err := enc.Encode(raw); err != nil {
|
||
|
s.Fini()
|
||
|
log.Fatal(err)
|
||
|
}
|
||
|
|
||
|
lock := true
|
||
|
displayHelloWorld(s)
|
||
|
displaySixel(s, img, lock)
|
||
|
|
||
|
for {
|
||
|
switch ev := s.PollEvent().(type) {
|
||
|
case *tcell.EventResize:
|
||
|
s.Sync()
|
||
|
displayHelloWorld(s)
|
||
|
displaySixel(s, img, lock)
|
||
|
case *tcell.EventKey:
|
||
|
if ev.Key() == tcell.KeyEscape {
|
||
|
s.Fini()
|
||
|
os.Exit(0)
|
||
|
}
|
||
|
if ev.Key() == tcell.KeyEnter {
|
||
|
lock = !lock
|
||
|
displaySixel(s, img, lock)
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|