mirror of
https://github.com/mrusme/neonmodem.git
synced 2024-12-04 14:46:37 -05:00
Refactored event/msg handling
This commit is contained in:
parent
3587cb9ccc
commit
740c69167e
96
ui/toolkit/msg.go
Normal file
96
ui/toolkit/msg.go
Normal file
@ -0,0 +1,96 @@
|
|||||||
|
package toolkit
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strconv"
|
||||||
|
|
||||||
|
"github.com/charmbracelet/bubbles/key"
|
||||||
|
tea "github.com/charmbracelet/bubbletea"
|
||||||
|
"github.com/mrusme/gobbs/ui/cmd"
|
||||||
|
)
|
||||||
|
|
||||||
|
type MsgHandlingKeymapKey struct {
|
||||||
|
ID string
|
||||||
|
Handler func(m interface{}) (bool, []tea.Cmd)
|
||||||
|
}
|
||||||
|
|
||||||
|
type MsgHandling struct {
|
||||||
|
OnKeymapKey []MsgHandlingKeymapKey
|
||||||
|
OnAnyNumberKey func(m interface{}, n int8) (bool, []tea.Cmd)
|
||||||
|
OnAnyUncaughtKey func(m interface{}, k tea.KeyMsg) (bool, []tea.Cmd)
|
||||||
|
OnViewResize func(m interface{}) (bool, []tea.Cmd)
|
||||||
|
OnWinOpenCmd func(m interface{}, c cmd.Command) (bool, []tea.Cmd)
|
||||||
|
OnWinRefreshDataCmd func(m interface{}, c cmd.Command) (bool, []tea.Cmd)
|
||||||
|
OnWinFreshDataCmd func(m interface{}, c cmd.Command) (bool, []tea.Cmd)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (tk *ToolKit) SetMsgHandling(mh MsgHandling) {
|
||||||
|
tk.mh = mh
|
||||||
|
}
|
||||||
|
|
||||||
|
func (tk *ToolKit) HandleMsg(m interface{}, msg tea.Msg) (bool, []tea.Cmd) {
|
||||||
|
var cmds []tea.Cmd
|
||||||
|
|
||||||
|
switch msg := msg.(type) {
|
||||||
|
case tea.KeyMsg:
|
||||||
|
for i := 0; i < len(tk.mh.OnKeymapKey); i++ {
|
||||||
|
if key.Matches(msg, tk.KeymapGet(tk.mh.OnKeymapKey[i].ID)) {
|
||||||
|
return tk.mh.OnKeymapKey[i].Handler(m)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if tk.mh.OnAnyNumberKey != nil {
|
||||||
|
switch msg.String() {
|
||||||
|
case "1", "2", "3", "4", "5", "6", "7", "8", "9", "0":
|
||||||
|
n, _ := strconv.Atoi(msg.String())
|
||||||
|
return tk.mh.OnAnyNumberKey(m, int8(n))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if tk.mh.OnAnyUncaughtKey != nil {
|
||||||
|
return tk.mh.OnAnyUncaughtKey(m, msg)
|
||||||
|
}
|
||||||
|
|
||||||
|
case tea.WindowSizeMsg:
|
||||||
|
tk.wh[0] = msg.Width
|
||||||
|
tk.wh[1] = msg.Height
|
||||||
|
if tk.mh.OnViewResize != nil {
|
||||||
|
return tk.mh.OnViewResize(m)
|
||||||
|
}
|
||||||
|
return false, cmds
|
||||||
|
|
||||||
|
case cmd.Command:
|
||||||
|
tk.logger.Debugf("got command: %v\n", msg)
|
||||||
|
switch msg.Call {
|
||||||
|
case cmd.WinFocus:
|
||||||
|
if msg.Target == tk.winID ||
|
||||||
|
msg.Target == "*" {
|
||||||
|
tk.logger.Debug("got WinFocus")
|
||||||
|
tk.Focus(m)
|
||||||
|
}
|
||||||
|
tk.logger.Debugf("focused: %v", tk.focused)
|
||||||
|
return true, nil
|
||||||
|
case cmd.WinBlur:
|
||||||
|
if msg.Target == tk.winID ||
|
||||||
|
msg.Target == "*" {
|
||||||
|
tk.logger.Debug("got WinBlur")
|
||||||
|
tk.Blur(m)
|
||||||
|
}
|
||||||
|
tk.logger.Debugf("focused: %v", tk.focused)
|
||||||
|
return true, nil
|
||||||
|
case cmd.WinOpen:
|
||||||
|
if tk.mh.OnWinOpenCmd != nil {
|
||||||
|
return tk.mh.OnWinOpenCmd(m, msg)
|
||||||
|
}
|
||||||
|
case cmd.WinRefreshData:
|
||||||
|
if tk.mh.OnWinRefreshDataCmd != nil {
|
||||||
|
return tk.mh.OnWinRefreshDataCmd(m, msg)
|
||||||
|
}
|
||||||
|
case cmd.WinFreshData:
|
||||||
|
if tk.mh.OnWinFreshDataCmd != nil {
|
||||||
|
return tk.mh.OnWinFreshDataCmd(m, msg)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
return false, cmds
|
||||||
|
}
|
@ -2,8 +2,6 @@ package toolkit
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/charmbracelet/bubbles/key"
|
"github.com/charmbracelet/bubbles/key"
|
||||||
tea "github.com/charmbracelet/bubbletea"
|
|
||||||
"github.com/mrusme/gobbs/ui/cmd"
|
|
||||||
"github.com/mrusme/gobbs/ui/theme"
|
"github.com/mrusme/gobbs/ui/theme"
|
||||||
"go.uber.org/zap"
|
"go.uber.org/zap"
|
||||||
)
|
)
|
||||||
@ -15,6 +13,8 @@ type ToolKit struct {
|
|||||||
theme *theme.Theme
|
theme *theme.Theme
|
||||||
logger *zap.SugaredLogger
|
logger *zap.SugaredLogger
|
||||||
|
|
||||||
|
mh MsgHandling
|
||||||
|
|
||||||
m interface{}
|
m interface{}
|
||||||
wh [2]int
|
wh [2]int
|
||||||
focused bool
|
focused bool
|
||||||
@ -31,6 +31,8 @@ func New(winID string, t *theme.Theme, l *zap.SugaredLogger) *ToolKit {
|
|||||||
tk.theme = t
|
tk.theme = t
|
||||||
tk.logger = l
|
tk.logger = l
|
||||||
|
|
||||||
|
tk.mh = MsgHandling{}
|
||||||
|
|
||||||
tk.wh = [2]int{0, 0}
|
tk.wh = [2]int{0, 0}
|
||||||
tk.focused = false
|
tk.focused = false
|
||||||
|
|
||||||
@ -90,37 +92,3 @@ func (tk *ToolKit) ViewWidth() int {
|
|||||||
func (tk *ToolKit) ViewHeight() int {
|
func (tk *ToolKit) ViewHeight() int {
|
||||||
return tk.wh[1]
|
return tk.wh[1]
|
||||||
}
|
}
|
||||||
|
|
||||||
func (tk *ToolKit) HandleMsg(m interface{}, msg tea.Msg) (bool, []tea.Cmd) {
|
|
||||||
var cmds []tea.Cmd
|
|
||||||
|
|
||||||
switch msg := msg.(type) {
|
|
||||||
case tea.WindowSizeMsg:
|
|
||||||
tk.wh[0] = msg.Width
|
|
||||||
tk.wh[1] = msg.Height
|
|
||||||
return false, cmds
|
|
||||||
|
|
||||||
case cmd.Command:
|
|
||||||
tk.logger.Debugf("got command: %v\n", msg)
|
|
||||||
switch msg.Call {
|
|
||||||
case cmd.WinFocus:
|
|
||||||
if msg.Target == tk.winID ||
|
|
||||||
msg.Target == "*" {
|
|
||||||
tk.logger.Debug("got WinFocus")
|
|
||||||
tk.Focus(m)
|
|
||||||
}
|
|
||||||
tk.logger.Debugf("focused: %v", tk.focused)
|
|
||||||
return true, nil
|
|
||||||
case cmd.WinBlur:
|
|
||||||
if msg.Target == tk.winID ||
|
|
||||||
msg.Target == "*" {
|
|
||||||
tk.logger.Debug("got WinBlur")
|
|
||||||
tk.Blur(m)
|
|
||||||
}
|
|
||||||
tk.logger.Debugf("focused: %v", tk.focused)
|
|
||||||
return true, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
return false, cmds
|
|
||||||
}
|
|
||||||
|
@ -5,7 +5,6 @@ import (
|
|||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/charmbracelet/bubbles/key"
|
|
||||||
"github.com/charmbracelet/bubbles/viewport"
|
"github.com/charmbracelet/bubbles/viewport"
|
||||||
tea "github.com/charmbracelet/bubbletea"
|
tea "github.com/charmbracelet/bubbletea"
|
||||||
"github.com/charmbracelet/glamour"
|
"github.com/charmbracelet/glamour"
|
||||||
@ -69,9 +68,139 @@ func NewModel(c *ctx.Ctx) Model {
|
|||||||
|
|
||||||
m.tk.KeymapAdd("reply", "reply", "r")
|
m.tk.KeymapAdd("reply", "reply", "r")
|
||||||
|
|
||||||
|
m.tk.SetMsgHandling(toolkit.MsgHandling{
|
||||||
|
OnKeymapKey: []toolkit.MsgHandlingKeymapKey{
|
||||||
|
{
|
||||||
|
ID: "reply",
|
||||||
|
Handler: handleReply,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
OnAnyNumberKey: handleNumberKeys,
|
||||||
|
OnAnyUncaughtKey: handleUncaughtKeys,
|
||||||
|
OnViewResize: handleViewResize,
|
||||||
|
OnWinOpenCmd: handleWinOpenCmd,
|
||||||
|
OnWinRefreshDataCmd: handleWinOpenCmd,
|
||||||
|
OnWinFreshDataCmd: handleWinFreshDataCmd,
|
||||||
|
})
|
||||||
|
|
||||||
return m
|
return m
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func handleReply(mi interface{}) (bool, []tea.Cmd) {
|
||||||
|
var m *Model = mi.(*Model)
|
||||||
|
var cmds []tea.Cmd
|
||||||
|
var replyToIdx int = 0
|
||||||
|
var err error
|
||||||
|
|
||||||
|
// m.viewcache = m.buildView(false)
|
||||||
|
m.tk.CacheView(m)
|
||||||
|
|
||||||
|
if m.buffer != "" {
|
||||||
|
replyToIdx, err = strconv.Atoi(m.buffer)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
// TODO: Handle error
|
||||||
|
}
|
||||||
|
|
||||||
|
if replyToIdx >= len(m.replyIDs) {
|
||||||
|
// TODO: Handle error
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
m.ctx.Logger.Debugf("replyToIdx: %d", replyToIdx)
|
||||||
|
var rtype cmd.Arg = cmd.Arg{Name: "replyTo"}
|
||||||
|
var rarg cmd.Arg
|
||||||
|
var ridx cmd.Arg = cmd.Arg{Name: "replyToIdx", Value: replyToIdx}
|
||||||
|
|
||||||
|
if replyToIdx == 0 {
|
||||||
|
rtype.Value = "post"
|
||||||
|
rarg.Name = "post"
|
||||||
|
rarg.Value = *m.activePost
|
||||||
|
} else {
|
||||||
|
rtype.Value = "reply"
|
||||||
|
rarg.Name = "reply"
|
||||||
|
rarg.Value = *m.allReplies[(replyToIdx - 1)]
|
||||||
|
}
|
||||||
|
|
||||||
|
cmd := cmd.New(cmd.WinOpen, postcreate.WIN_ID, rtype, rarg, ridx)
|
||||||
|
cmds = append(cmds, cmd.Tea())
|
||||||
|
|
||||||
|
m.ctx.Logger.Debugln("caching view")
|
||||||
|
m.ctx.Logger.Debugf("buffer: %s", m.buffer)
|
||||||
|
// m.viewcache = m.buildView(false)
|
||||||
|
|
||||||
|
return true, cmds
|
||||||
|
}
|
||||||
|
|
||||||
|
func handleNumberKeys(mi interface{}, n int8) (bool, []tea.Cmd) {
|
||||||
|
var m *Model = mi.(*Model)
|
||||||
|
var cmds []tea.Cmd
|
||||||
|
|
||||||
|
m.buffer += strconv.Itoa(int(n))
|
||||||
|
|
||||||
|
return false, cmds
|
||||||
|
}
|
||||||
|
|
||||||
|
func handleUncaughtKeys(mi interface{}, k tea.KeyMsg) (bool, []tea.Cmd) {
|
||||||
|
var m *Model = mi.(*Model)
|
||||||
|
var cmds []tea.Cmd
|
||||||
|
|
||||||
|
m.buffer = ""
|
||||||
|
|
||||||
|
return false, cmds
|
||||||
|
}
|
||||||
|
|
||||||
|
func handleViewResize(mi interface{}) (bool, []tea.Cmd) {
|
||||||
|
var m *Model = mi.(*Model)
|
||||||
|
var cmds []tea.Cmd
|
||||||
|
|
||||||
|
m.ctx.Logger.Debugf("received WindowSizeMsg: %vx%v\n", m.tk.ViewWidth(), m.tk.ViewHeight())
|
||||||
|
viewportWidth := m.tk.ViewWidth() - 2
|
||||||
|
viewportHeight := m.tk.ViewHeight() - 5
|
||||||
|
|
||||||
|
viewportStyle.Width(viewportWidth)
|
||||||
|
viewportStyle.Height(viewportHeight)
|
||||||
|
m.viewport = viewport.New(viewportWidth-4, viewportHeight-4)
|
||||||
|
m.viewport.Width = viewportWidth - 4
|
||||||
|
m.viewport.Height = viewportHeight + 1
|
||||||
|
// cmds = append(cmds, viewport.Sync(m.viewport))
|
||||||
|
|
||||||
|
return false, cmds
|
||||||
|
}
|
||||||
|
|
||||||
|
func handleWinOpenCmd(mi interface{}, c cmd.Command) (bool, []tea.Cmd) {
|
||||||
|
var m *Model = mi.(*Model)
|
||||||
|
var cmds []tea.Cmd
|
||||||
|
|
||||||
|
if c.Target == WIN_ID {
|
||||||
|
m.ctx.Logger.Debug("got own WinOpen command")
|
||||||
|
m.activePost = c.GetArg("post").(*post.Post)
|
||||||
|
m.viewport.SetContent(m.renderViewport(m.activePost))
|
||||||
|
m.ctx.Logger.Debugf("loading post: %v", m.activePost.ID)
|
||||||
|
m.ctx.Loading = true
|
||||||
|
cmds = append(cmds, m.loadPost(m.activePost))
|
||||||
|
return true, cmds
|
||||||
|
}
|
||||||
|
|
||||||
|
return false, cmds
|
||||||
|
}
|
||||||
|
|
||||||
|
func handleWinFreshDataCmd(mi interface{}, c cmd.Command) (bool, []tea.Cmd) {
|
||||||
|
var m *Model = mi.(*Model)
|
||||||
|
var cmds []tea.Cmd
|
||||||
|
|
||||||
|
if c.Target == WIN_ID ||
|
||||||
|
c.Target == "*" {
|
||||||
|
m.ctx.Logger.Debug("got *post.Post")
|
||||||
|
m.activePost = c.GetArg("post").(*post.Post)
|
||||||
|
m.viewport.SetContent(m.renderViewport(m.activePost))
|
||||||
|
m.ctx.Loading = false
|
||||||
|
return true, cmds
|
||||||
|
}
|
||||||
|
|
||||||
|
return false, cmds
|
||||||
|
}
|
||||||
|
|
||||||
func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
||||||
var cmds []tea.Cmd
|
var cmds []tea.Cmd
|
||||||
|
|
||||||
@ -80,105 +209,6 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
return m, tea.Batch(cmds...)
|
return m, tea.Batch(cmds...)
|
||||||
}
|
}
|
||||||
|
|
||||||
switch msg := msg.(type) {
|
|
||||||
case tea.KeyMsg:
|
|
||||||
switch {
|
|
||||||
|
|
||||||
case key.Matches(msg, m.tk.KeymapGet("reply")):
|
|
||||||
var replyToIdx int = 0
|
|
||||||
var err error
|
|
||||||
|
|
||||||
// m.viewcache = m.buildView(false)
|
|
||||||
m.tk.CacheView(&m)
|
|
||||||
|
|
||||||
if m.buffer != "" {
|
|
||||||
replyToIdx, err = strconv.Atoi(m.buffer)
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
// TODO: Handle error
|
|
||||||
}
|
|
||||||
|
|
||||||
if replyToIdx >= len(m.replyIDs) {
|
|
||||||
// TODO: Handle error
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
m.ctx.Logger.Debugf("replyToIdx: %d", replyToIdx)
|
|
||||||
var rtype cmd.Arg = cmd.Arg{Name: "replyTo"}
|
|
||||||
var rarg cmd.Arg
|
|
||||||
var ridx cmd.Arg = cmd.Arg{Name: "replyToIdx", Value: replyToIdx}
|
|
||||||
|
|
||||||
if replyToIdx == 0 {
|
|
||||||
rtype.Value = "post"
|
|
||||||
rarg.Name = "post"
|
|
||||||
rarg.Value = *m.activePost
|
|
||||||
} else {
|
|
||||||
rtype.Value = "reply"
|
|
||||||
rarg.Name = "reply"
|
|
||||||
rarg.Value = *m.allReplies[(replyToIdx - 1)]
|
|
||||||
}
|
|
||||||
|
|
||||||
cmd := cmd.New(cmd.WinOpen, postcreate.WIN_ID, rtype, rarg, ridx)
|
|
||||||
cmds = append(cmds, cmd.Tea())
|
|
||||||
|
|
||||||
m.ctx.Logger.Debugln("caching view")
|
|
||||||
m.ctx.Logger.Debugf("buffer: %s", m.buffer)
|
|
||||||
// m.viewcache = m.buildView(false)
|
|
||||||
|
|
||||||
return m, tea.Batch(cmds...)
|
|
||||||
|
|
||||||
default:
|
|
||||||
switch msg.String() {
|
|
||||||
case "1", "2", "3", "4", "5", "6", "7", "8", "9", "0":
|
|
||||||
m.buffer += msg.String()
|
|
||||||
return m, nil
|
|
||||||
default:
|
|
||||||
m.buffer = ""
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
case tea.WindowSizeMsg:
|
|
||||||
m.ctx.Logger.Debugf("received WindowSizeMsg: %vx%v\n", m.tk.ViewWidth(), m.tk.ViewHeight())
|
|
||||||
viewportWidth := m.tk.ViewWidth() - 2
|
|
||||||
viewportHeight := m.tk.ViewHeight() - 5
|
|
||||||
|
|
||||||
viewportStyle.Width(viewportWidth)
|
|
||||||
viewportStyle.Height(viewportHeight)
|
|
||||||
m.viewport = viewport.New(viewportWidth-4, viewportHeight-4)
|
|
||||||
m.viewport.Width = viewportWidth - 4
|
|
||||||
m.viewport.Height = viewportHeight + 1
|
|
||||||
// cmds = append(cmds, viewport.Sync(m.viewport))
|
|
||||||
|
|
||||||
case cmd.Command:
|
|
||||||
m.ctx.Logger.Debugf("got command: %v\n", msg)
|
|
||||||
switch msg.Call {
|
|
||||||
case cmd.WinOpen, cmd.WinRefreshData:
|
|
||||||
if msg.Target == WIN_ID {
|
|
||||||
m.ctx.Logger.Debug("got own WinOpen command")
|
|
||||||
m.activePost = msg.GetArg("post").(*post.Post)
|
|
||||||
m.viewport.SetContent(m.renderViewport(m.activePost))
|
|
||||||
m.ctx.Logger.Debugf("loading post: %v", m.activePost.ID)
|
|
||||||
m.ctx.Loading = true
|
|
||||||
return m, m.loadPost(m.activePost)
|
|
||||||
}
|
|
||||||
return m, nil
|
|
||||||
case cmd.WinFreshData:
|
|
||||||
if msg.Target == WIN_ID ||
|
|
||||||
msg.Target == "*" {
|
|
||||||
m.ctx.Logger.Debug("got *post.Post")
|
|
||||||
m.activePost = msg.GetArg("post").(*post.Post)
|
|
||||||
m.viewport.SetContent(m.renderViewport(m.activePost))
|
|
||||||
m.ctx.Loading = false
|
|
||||||
return m, nil
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
m.ctx.Logger.Debugf("received unhandled command: %v\n", msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
default:
|
|
||||||
m.ctx.Logger.Debugf("received unhandled msg: %v\n", msg)
|
|
||||||
}
|
|
||||||
|
|
||||||
var cmd tea.Cmd
|
var cmd tea.Cmd
|
||||||
|
|
||||||
m.viewport, cmd = m.viewport.Update(msg)
|
m.viewport, cmd = m.viewport.Update(msg)
|
||||||
|
Loading…
Reference in New Issue
Block a user