mirror of
https://github.com/hibiken/asynq.git
synced 2025-10-03 05:12:01 +08:00
(cli): Add modal
This commit is contained in:
@@ -40,6 +40,7 @@ type State struct {
|
||||
|
||||
selectedQueue *asynq.QueueInfo // queue shown on queue details view
|
||||
selectedGroup *asynq.GroupInfo
|
||||
selectedTask *asynq.TaskInfo
|
||||
|
||||
pageNum int // pagination page number
|
||||
|
||||
|
@@ -49,6 +49,7 @@ func (dd *dashDrawer) draw(state *State) {
|
||||
drawTaskStateBreakdown(d, baseStyle, state)
|
||||
d.NL()
|
||||
drawTaskTable(d, state)
|
||||
drawTaskModal(d, state)
|
||||
case viewTypeServers:
|
||||
d.Println("=== Servers ===", baseStyle.Bold(true))
|
||||
d.NL()
|
||||
@@ -439,3 +440,74 @@ func drawTaskStateBreakdown(d *ScreenDrawer, style tcell.Style, state *State) {
|
||||
}
|
||||
d.NL()
|
||||
}
|
||||
|
||||
func drawTaskModal(d *ScreenDrawer, state *State) {
|
||||
contents := []*rowContent{
|
||||
{" === Task Summary ===", baseStyle.Bold(true)},
|
||||
{"", baseStyle},
|
||||
{" ID: xxxxx", baseStyle},
|
||||
{" Type: xxxxx", baseStyle},
|
||||
{" State: xxxx", baseStyle},
|
||||
}
|
||||
withModal(d, contents)
|
||||
}
|
||||
|
||||
type rowContent struct {
|
||||
s string // should not include newline
|
||||
style tcell.Style
|
||||
}
|
||||
|
||||
func withModal(d *ScreenDrawer, contents []*rowContent) {
|
||||
modalStyle := baseStyle //.Background(tcell.ColorDarkMagenta)
|
||||
w, h := d.Screen().Size()
|
||||
var (
|
||||
modalWidth = int(math.Floor(float64(w) * 0.6))
|
||||
modalHeight = int(math.Floor(float64(h) * 0.6))
|
||||
rowOffset = int(math.Floor(float64(h) * 0.2)) // 20% from the top
|
||||
colOffset = int(math.Floor(float64(w) * 0.2)) // 20% from the left
|
||||
)
|
||||
if modalHeight < 3 {
|
||||
return // no content can be shown
|
||||
}
|
||||
d.Goto(colOffset, rowOffset)
|
||||
d.Print(string(tcell.RuneULCorner), modalStyle)
|
||||
d.Print(strings.Repeat(string(tcell.RuneHLine), modalWidth-2), modalStyle)
|
||||
d.Print(string(tcell.RuneURCorner), modalStyle)
|
||||
d.NL()
|
||||
for i := 1; i < modalHeight-1; i++ {
|
||||
d.Goto(colOffset, rowOffset+i)
|
||||
d.Print(string(tcell.RuneVLine), modalStyle)
|
||||
cnt := &rowContent{strings.Repeat(" ", modalWidth-2), baseStyle}
|
||||
if i <= len(contents) {
|
||||
cnt = contents[i-1]
|
||||
cnt.s = adjustWidth(cnt.s, modalWidth-2)
|
||||
}
|
||||
d.Print(truncate(cnt.s, modalWidth-2), cnt.style)
|
||||
d.Print(string(tcell.RuneVLine), modalStyle)
|
||||
d.NL()
|
||||
}
|
||||
d.Goto(colOffset, rowOffset+modalHeight-1)
|
||||
d.Print(string(tcell.RuneLLCorner), modalStyle)
|
||||
d.Print(strings.Repeat(string(tcell.RuneHLine), modalWidth-2), modalStyle)
|
||||
d.Print(string(tcell.RuneLRCorner), modalStyle)
|
||||
d.NL()
|
||||
}
|
||||
|
||||
func adjustWidth(s string, width int) string {
|
||||
sw := runewidth.StringWidth(s)
|
||||
if sw > width {
|
||||
return truncate(s, width)
|
||||
}
|
||||
var b strings.Builder
|
||||
b.WriteString(s)
|
||||
b.WriteString(strings.Repeat(" ", width-sw))
|
||||
return b.String()
|
||||
}
|
||||
|
||||
// truncates s if s exceeds max length.
|
||||
func truncate(s string, max int) string {
|
||||
if runewidth.StringWidth(s) <= max {
|
||||
return s
|
||||
}
|
||||
return string([]rune(s)[:max-1]) + "…"
|
||||
}
|
||||
|
33
tools/asynq/cmd/dash/draw_test.go
Normal file
33
tools/asynq/cmd/dash/draw_test.go
Normal file
@@ -0,0 +1,33 @@
|
||||
// Copyright 2022 Kentaro Hibino. All rights reserved.
|
||||
// Use of this source code is governed by a MIT license
|
||||
// that can be found in the LICENSE file.
|
||||
|
||||
package dash
|
||||
|
||||
import "testing"
|
||||
|
||||
func TestTruncate(t *testing.T) {
|
||||
tests := []struct {
|
||||
s string
|
||||
max int
|
||||
want string
|
||||
}{
|
||||
{
|
||||
s: "hello world!",
|
||||
max: 15,
|
||||
want: "hello world!",
|
||||
},
|
||||
{
|
||||
s: "hello world!",
|
||||
max: 6,
|
||||
want: "hello…",
|
||||
},
|
||||
}
|
||||
|
||||
for _, tc := range tests {
|
||||
got := truncate(tc.s, tc.max)
|
||||
if tc.want != got {
|
||||
t.Errorf("truncate(%q, %d) = %q, want %q", tc.s, tc.max, got, tc.want)
|
||||
}
|
||||
}
|
||||
}
|
@@ -57,6 +57,12 @@ func (d *ScreenDrawer) Screen() tcell.Screen {
|
||||
return d.l.s
|
||||
}
|
||||
|
||||
// Goto moves the screendrawer to the specified cell.
|
||||
func (d *ScreenDrawer) Goto(x, y int) {
|
||||
d.l.row = y
|
||||
d.l.col = x
|
||||
}
|
||||
|
||||
// Go to the bottom of the screen.
|
||||
func (d *ScreenDrawer) GoToBottom() {
|
||||
_, h := d.Screen().Size()
|
||||
|
Reference in New Issue
Block a user