1
0
mirror of https://github.com/mrusme/neonmodem.git synced 2024-12-04 14:46:37 -05:00
neonmodem/ui/views/posts/posts.go
2024-07-31 17:08:14 -05:00

279 lines
5.8 KiB
Go

package posts
import (
"errors"
"strings"
"github.com/charmbracelet/bubbles/key"
"github.com/charmbracelet/bubbles/list"
tea "github.com/charmbracelet/bubbletea"
"github.com/charmbracelet/lipgloss"
"github.com/mrusme/neonmodem/aggregator"
"github.com/mrusme/neonmodem/models/post"
"github.com/mrusme/neonmodem/ui/cmd"
"github.com/mrusme/neonmodem/ui/ctx"
"github.com/mrusme/neonmodem/ui/windows/postcreate"
"github.com/mrusme/neonmodem/ui/windows/postshow"
)
var (
VIEW_ID = "posts"
)
type KeyMap struct {
Refresh key.Binding
NewPost key.Binding
Select key.Binding
Quit key.Binding
}
var DefaultKeyMap = KeyMap{
Refresh: key.NewBinding(
key.WithKeys("ctrl+r"),
key.WithHelp("ctrl+r", "refresh"),
),
NewPost: key.NewBinding(
key.WithKeys("n"),
key.WithHelp("n", "new post"),
),
Select: key.NewBinding(
key.WithKeys("r", "enter"),
key.WithHelp("r/enter", "read"),
),
Quit: key.NewBinding(
key.WithKeys("esc"),
key.WithHelp("esc", "quit"),
),
}
type Model struct {
ctx *ctx.Ctx
keymap KeyMap
focused bool
list list.Model
items []list.Item
a *aggregator.Aggregator
viewcache string
viewcacheTextareaXY []int
}
func (m Model) Init() tea.Cmd {
return nil
}
func NewModel(c *ctx.Ctx) Model {
m := Model{
ctx: c,
keymap: DefaultKeyMap,
focused: false,
viewcache: "",
viewcacheTextareaXY: []int{0, 0, 0, 0},
}
listDelegate := list.NewDefaultDelegate()
listDelegate.Styles.NormalTitle = m.ctx.Theme.PostsList.Item.Focused
listDelegate.Styles.DimmedTitle = m.ctx.Theme.PostsList.Item.Blurred
listDelegate.Styles.SelectedTitle = m.ctx.Theme.PostsList.Item.Selected
listDelegate.Styles.NormalDesc = m.ctx.Theme.PostsList.ItemDetail.Focused
listDelegate.Styles.DimmedDesc = m.ctx.Theme.PostsList.ItemDetail.Blurred
listDelegate.Styles.SelectedDesc = m.ctx.Theme.PostsList.ItemDetail.Selected
m.list = list.New(m.items, listDelegate, 0, 0)
m.list.SetShowTitle(false)
m.list.SetShowStatusBar(false)
m.list.DisableQuitKeybindings()
m.a, _ = aggregator.New(m.ctx)
return m
}
func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
var cmds []tea.Cmd
switch msg := msg.(type) {
case tea.KeyMsg:
switch {
case key.Matches(msg, m.keymap.Quit):
if m.list.FilterState() == list.Filtering {
break
}
return m, tea.Quit
case key.Matches(msg, m.keymap.Refresh):
m.ctx.Loading = true
cmds = append(cmds, m.refresh())
case key.Matches(msg, m.keymap.Select):
i, ok := m.list.SelectedItem().(post.Post)
if ok {
m.viewcache = m.buildView(false)
cmd := cmd.New(cmd.WinOpen, postshow.WIN_ID, cmd.Arg{
Name: "post",
Value: &i,
})
cmds = append(cmds, cmd.Tea())
}
case key.Matches(msg, m.keymap.NewPost):
if m.list.FilterState() == list.Filtering {
break
}
i, ok := m.list.SelectedItem().(post.Post)
if ok {
caps := (*m.ctx.Systems[i.SysIDX]).GetCapabilities()
if !caps.IsCapableOf("create:post") {
cmds = append(cmds, cmd.New(
cmd.MsgError,
VIEW_ID,
cmd.Arg{
Name: "error",
Value: errors.New(
"This system doesn't support posting yet!\n",
),
},
).Tea())
return m, tea.Batch(cmds...)
}
m.focused = false // TODO: Refactor and use ToolKit
m.viewcache = m.buildView(false)
cmd := cmd.New(
cmd.WinOpen,
postcreate.WIN_ID,
cmd.Arg{
Name: "action",
Value: "post",
},
cmd.Arg{
Name: "post",
Value: &i,
},
)
cmds = append(cmds, cmd.Tea())
return m, tea.Batch(cmds...)
}
}
case tea.WindowSizeMsg:
listWidth := m.ctx.Content[0] - 2
listHeight := m.ctx.Content[1] - 1
m.ctx.Theme.PostsList.List.Focused.Width(listWidth)
m.ctx.Theme.PostsList.List.Blurred.Width(listWidth)
m.ctx.Theme.PostsList.List.Focused.Height(listHeight)
m.ctx.Theme.PostsList.List.Blurred.Height(listHeight)
m.list.SetSize(
listWidth-2,
listHeight-2,
)
msg.Width = listWidth
msg.Height = listHeight
case cmd.Command:
switch msg.Call {
case cmd.ViewFocus:
if msg.Target == VIEW_ID ||
msg.Target == "*" {
m.Focus()
}
return m, nil
case cmd.ViewBlur:
if msg.Target == VIEW_ID ||
msg.Target == "*" {
m.Blur()
}
return m, nil
case cmd.ViewRefreshData:
if msg.Target == VIEW_ID ||
msg.Target == "*" {
m.ctx.Loading = true
cmds = append(cmds, m.refresh())
}
case cmd.ViewFreshData:
if msg.Target == VIEW_ID ||
msg.Target == "*" {
m.items = msg.GetArg("items").([]list.Item)
m.list.SetItems(m.items)
m.ctx.Loading = false
return m, nil
}
}
}
var lcmd tea.Cmd
m.list, lcmd = m.list.Update(msg)
cmds = append(cmds, lcmd)
return m, tea.Batch(cmds...)
}
func (m *Model) refresh() tea.Cmd {
return func() tea.Msg {
var items []list.Item
posts, errs := m.a.ListPosts()
for _, err := range errs {
if err != nil {
m.ctx.Logger.Error(errs)
// ccmds = append(ccmds, cmd.New(
// cmd.MsgError,
// VIEW_ID,
// cmd.Arg{Name: "errors", Value: errs},
// ).Tea())
}
}
for _, post := range posts {
items = append(items, post)
}
return *cmd.New(
cmd.ViewFreshData,
VIEW_ID,
cmd.Arg{Name: "items", Value: items},
)
}
}
func (m *Model) Focus() {
m.focused = true
m.viewcache = m.buildView(false)
}
func (m *Model) Blur() {
m.focused = false
m.viewcache = m.buildView(false)
}
func (m Model) View() string {
return m.buildView(true)
}
func (m Model) buildView(cached bool) string {
var view strings.Builder = strings.Builder{}
if cached && m.focused == false && m.viewcache != "" {
return m.viewcache
}
var l string = ""
if m.focused {
l = m.ctx.Theme.PostsList.List.Focused.Render(m.list.View())
} else {
l = m.ctx.Theme.PostsList.List.Blurred.Render(m.list.View())
}
view.WriteString(lipgloss.JoinHorizontal(
lipgloss.Top,
l,
))
m.viewcache = view.String()
return m.viewcache
}