1
0
mirror of https://github.com/mrusme/neonmodem.git synced 2024-11-03 04:27:16 -05:00
neonmodem/ui/ui.go

276 lines
6.1 KiB
Go
Raw Normal View History

2022-12-28 22:22:36 -05:00
package ui
import (
// "fmt"
"strings"
2023-01-04 21:43:11 -05:00
"github.com/mrusme/gobbs/models/forum"
"github.com/mrusme/gobbs/system"
"github.com/mrusme/gobbs/ui/cmd"
2022-12-28 22:22:36 -05:00
"github.com/mrusme/gobbs/ui/ctx"
"github.com/mrusme/gobbs/ui/header"
2022-12-28 22:27:46 -05:00
"github.com/mrusme/gobbs/ui/views/posts"
"github.com/mrusme/gobbs/ui/windowmanager"
2023-01-03 00:32:48 -05:00
"github.com/mrusme/gobbs/ui/windows/msgerror"
2023-01-04 21:43:11 -05:00
"github.com/mrusme/gobbs/ui/windows/popuplist"
2023-01-02 18:46:26 -05:00
"github.com/mrusme/gobbs/ui/windows/postcreate"
2023-01-02 15:33:12 -05:00
"github.com/mrusme/gobbs/ui/windows/postshow"
2022-12-28 22:22:36 -05:00
"github.com/mrusme/gobbs/ui/views"
"github.com/charmbracelet/bubbles/key"
2023-01-04 21:43:11 -05:00
"github.com/charmbracelet/bubbles/list"
2023-01-02 15:29:29 -05:00
"github.com/charmbracelet/bubbles/spinner"
2022-12-28 22:22:36 -05:00
tea "github.com/charmbracelet/bubbletea"
)
type KeyMap struct {
2023-01-04 21:43:11 -05:00
SystemSelect key.Binding
ForumSelect key.Binding
Close key.Binding
2022-12-28 22:22:36 -05:00
}
var DefaultKeyMap = KeyMap{
2023-01-04 21:43:11 -05:00
SystemSelect: key.NewBinding(
key.WithKeys("ctrl+s"),
key.WithHelp("C-s", "System selector"),
),
ForumSelect: key.NewBinding(
key.WithKeys("ctrl+f"),
key.WithHelp("C-f", "Forum selector"),
),
Close: key.NewBinding(
key.WithKeys("q", "esc"),
key.WithHelp("q/esc", "close"),
),
2022-12-28 22:22:36 -05:00
}
type Model struct {
keymap KeyMap
header header.Model
views []views.View
currentView int
wm *windowmanager.WM
ctx *ctx.Ctx
2023-01-02 19:27:23 -05:00
viewcache string
2023-01-03 13:19:21 -05:00
viewcacheID string
2023-01-02 19:27:23 -05:00
renderOnlyFocused bool
2022-12-28 22:22:36 -05:00
}
func NewModel(c *ctx.Ctx) Model {
m := Model{
keymap: DefaultKeyMap,
currentView: 0,
2023-01-02 16:45:42 -05:00
wm: windowmanager.New(c),
ctx: c,
2023-01-02 19:27:23 -05:00
viewcache: "",
renderOnlyFocused: false,
2022-12-28 22:22:36 -05:00
}
m.header = header.NewModel(m.ctx)
2022-12-31 19:06:39 -05:00
m.views = append(m.views, posts.NewModel(m.ctx))
2022-12-28 22:22:36 -05:00
return m
}
func (m Model) Init() tea.Cmd {
return tea.Batch(
tea.EnterAltScreen,
cmd.New(cmd.ViewFocus, "*").Tea(),
cmd.New(cmd.ViewRefreshData, "*").Tea(),
)
2022-12-28 22:22:36 -05:00
}
func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
cmds := make([]tea.Cmd, 0)
2023-01-03 13:19:21 -05:00
m.viewcacheID = m.wm.Focused()
2022-12-28 22:22:36 -05:00
switch msg := msg.(type) {
2023-01-02 14:24:07 -05:00
2022-12-28 22:22:36 -05:00
case tea.KeyMsg:
switch {
case key.Matches(msg, m.keymap.Close):
m.ctx.Logger.Debug("close received")
2023-01-02 16:45:42 -05:00
closed, ccmds := m.wm.CloseFocused()
if !closed {
m.ctx.Logger.Debug("CloseFocused() was false, quitting")
return m, tea.Quit
}
2023-01-02 16:45:42 -05:00
return m, tea.Batch(ccmds...)
2023-01-04 21:43:11 -05:00
case key.Matches(msg, m.keymap.SystemSelect):
var listItems []list.Item
for _, sys := range m.ctx.Systems {
listItems = append(listItems, *sys)
}
ccmds := m.wm.Open(
popuplist.WIN_ID,
popuplist.NewModel(m.ctx),
[4]int{
int(m.ctx.Content[1] / 2),
int(m.ctx.Content[1] / 4),
int(m.ctx.Content[1] / 2),
int(m.ctx.Content[1] / 4),
},
cmd.New(
cmd.WinOpen,
popuplist.WIN_ID,
cmd.Arg{Name: "selectionID", Value: "system"},
cmd.Arg{Name: "items", Value: listItems},
),
)
return m, tea.Batch(ccmds...)
2023-01-02 16:45:42 -05:00
default:
if m.wm.GetNumberOpen() > 0 {
cmd := m.wm.Update(m.wm.Focused(), msg)
return m, cmd
}
2022-12-28 22:22:36 -05:00
}
case tea.WindowSizeMsg:
m.setSizes(msg.Width, msg.Height)
for i := range m.views {
v, cmd := m.views[i].Update(msg)
m.views[i] = v
cmds = append(cmds, cmd)
}
2023-01-02 16:45:42 -05:00
m.ctx.Logger.Debugf("resizing all: %v\n", m.ctx.Content)
ccmds := m.wm.ResizeAll(m.ctx.Content[0], m.ctx.Content[1])
cmds = append(cmds, ccmds...)
case cmd.Command:
var ccmds []tea.Cmd
switch msg.Call {
2023-01-02 20:34:39 -05:00
case cmd.WinOpen:
2023-01-02 18:46:26 -05:00
switch msg.Target {
case postshow.WIN_ID:
m.ctx.Logger.Debugln("received WinOpen")
ccmds = m.wm.Open(
msg.Target,
postshow.NewModel(m.ctx),
2023-01-03 13:28:48 -05:00
[4]int{
3,
1,
6,
4,
},
2023-01-02 18:46:26 -05:00
&msg,
)
case postcreate.WIN_ID:
m.ctx.Logger.Debugln("received WinOpen")
ccmds = m.wm.Open(
msg.Target,
postcreate.NewModel(m.ctx),
2023-01-03 13:28:48 -05:00
[4]int{
6,
m.ctx.Content[1] - 16,
10,
4,
},
2023-01-02 18:46:26 -05:00
&msg,
)
2023-01-03 13:19:21 -05:00
m.viewcache = m.buildView(false)
2023-01-02 18:46:26 -05:00
}
m.ctx.Logger.Debugf("got back ccmds: %v\n", ccmds)
2023-01-02 20:34:39 -05:00
2023-01-02 19:27:23 -05:00
case cmd.WinClose:
switch msg.Target {
case postcreate.WIN_ID:
m.ctx.Logger.Debugln("received WinClose")
2023-01-04 21:43:11 -05:00
case popuplist.WIN_ID:
switch msg.GetArg("selectionID").(string) {
case "system":
selected := msg.GetArg("selected").(system.System)
m.ctx.SetCurrentSystem(selected.GetID())
case "forum":
selected := msg.GetArg("selected").(forum.Forum)
m.ctx.SetCurrentForum(selected.ID)
}
2023-01-02 19:27:23 -05:00
}
2023-01-02 20:34:39 -05:00
case cmd.WMCloseWin:
2023-01-04 21:43:11 -05:00
if ok, clcmds := m.wm.Close(msg.Target, msg.GetArgs()...); ok {
2023-01-02 20:34:39 -05:00
cmds = append(cmds, clcmds...)
}
2023-01-03 00:32:48 -05:00
case cmd.MsgError:
m.ctx.Logger.Debugln("received MsgError")
ccmds = m.wm.Open(
msgerror.WIN_ID,
msgerror.NewModel(m.ctx),
2023-01-03 13:28:48 -05:00
[4]int{
int(m.ctx.Content[1] / 2),
int(m.ctx.Content[1] / 4),
int(m.ctx.Content[1] / 2),
int(m.ctx.Content[1] / 4),
},
2023-01-03 00:32:48 -05:00
&msg,
)
default:
2023-01-03 13:19:21 -05:00
// if msg.Call < cmd.ViewFocus {
m.ctx.Logger.Debugf("updating all with cmd: %v\n", msg)
ccmds = m.wm.UpdateAll(msg)
// }
}
cmds = append(cmds, ccmds...)
2023-01-02 14:24:07 -05:00
2023-01-02 15:29:29 -05:00
case spinner.TickMsg:
// Do nothing
2023-01-02 14:24:07 -05:00
default:
2023-01-02 18:46:26 -05:00
m.ctx.Logger.Debugf("updating focused with default: %v\n", msg)
cmds = append(cmds, m.wm.UpdateFocused(msg)...)
2022-12-28 22:22:36 -05:00
}
v, vcmd := m.views[m.currentView].Update(msg)
m.views[m.currentView] = v
cmds = append(cmds, vcmd)
2022-12-28 22:22:36 -05:00
header, hcmd := m.header.Update(msg)
m.header = header
cmds = append(cmds, hcmd)
2022-12-28 22:22:36 -05:00
return m, tea.Batch(cmds...)
}
func (m Model) View() string {
2023-01-02 19:27:23 -05:00
return m.buildView(true)
}
func (m Model) buildView(cached bool) string {
2022-12-28 22:22:36 -05:00
s := strings.Builder{}
2023-01-02 19:27:23 -05:00
var tmp string = ""
2023-01-03 13:19:21 -05:00
m.ctx.Logger.Debugf("viewcacheID: %s\n", m.viewcacheID)
if cached && m.viewcache != "" && m.viewcacheID == m.wm.Focused() &&
m.viewcacheID == postcreate.WIN_ID {
m.ctx.Logger.Debug("hitting UI viewcache")
2023-01-02 19:27:23 -05:00
tmp = m.viewcache
2023-01-03 13:19:21 -05:00
m.renderOnlyFocused = true
2023-01-02 19:27:23 -05:00
} else {
2023-01-03 13:19:21 -05:00
m.ctx.Logger.Debug("generating UI viewcache")
m.renderOnlyFocused = false
2023-01-02 19:27:23 -05:00
s.WriteString(m.header.View() + "\n")
s.WriteString(m.views[m.currentView].View())
tmp = s.String()
}
2023-01-02 19:27:23 -05:00
return m.wm.View(tmp, m.renderOnlyFocused)
2022-12-28 22:22:36 -05:00
}
func (m Model) setSizes(winWidth int, winHeight int) {
(*m.ctx).Screen[0] = winWidth
(*m.ctx).Screen[1] = winHeight
m.ctx.Content[0] = m.ctx.Screen[0]
m.ctx.Content[1] = m.ctx.Screen[1] - 8
2022-12-28 22:22:36 -05:00
}