mirror of
https://github.com/mrusme/neonmodem.git
synced 2024-12-04 14:46:37 -05:00
Enhanced UI implementation
This commit is contained in:
parent
50e683f0fd
commit
f608100050
@ -9,10 +9,12 @@ const (
|
|||||||
WinFocus
|
WinFocus
|
||||||
WinBlur
|
WinBlur
|
||||||
WinRefreshData
|
WinRefreshData
|
||||||
|
WinFreshData
|
||||||
|
|
||||||
ViewFocus
|
ViewFocus
|
||||||
ViewBlur
|
ViewBlur
|
||||||
ViewRefreshData
|
ViewRefreshData
|
||||||
|
ViewFreshData
|
||||||
)
|
)
|
||||||
|
|
||||||
type Arg struct {
|
type Arg struct {
|
||||||
|
13
ui/ui.go
13
ui/ui.go
@ -15,6 +15,7 @@ import (
|
|||||||
"github.com/mrusme/gobbs/ui/views"
|
"github.com/mrusme/gobbs/ui/views"
|
||||||
|
|
||||||
"github.com/charmbracelet/bubbles/key"
|
"github.com/charmbracelet/bubbles/key"
|
||||||
|
"github.com/charmbracelet/bubbles/spinner"
|
||||||
tea "github.com/charmbracelet/bubbletea"
|
tea "github.com/charmbracelet/bubbletea"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -109,20 +110,24 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
msg.Target,
|
msg.Target,
|
||||||
postdialog.NewModel(m.ctx),
|
postdialog.NewModel(m.ctx),
|
||||||
[4]int{3, 2, 10, 6},
|
[4]int{3, 2, 10, 6},
|
||||||
msg.GetArgs()...,
|
&msg,
|
||||||
)
|
)
|
||||||
m.ctx.Logger.Debugf("got back ccmds: %v\n", ccmds)
|
m.ctx.Logger.Debugf("got back ccmds: %v\n", ccmds)
|
||||||
default:
|
default:
|
||||||
m.ctx.Logger.Debugf("updating all with cmd: %v\n", msg)
|
if msg.Call < cmd.ViewFocus {
|
||||||
ccmds = m.wm.UpdateAll(msg)
|
m.ctx.Logger.Debugf("updating all with cmd: %v\n", msg)
|
||||||
|
ccmds = m.wm.UpdateAll(msg)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
cmds = append(cmds, ccmds...)
|
cmds = append(cmds, ccmds...)
|
||||||
|
|
||||||
|
case spinner.TickMsg:
|
||||||
|
// Do nothing
|
||||||
|
|
||||||
default:
|
default:
|
||||||
m.ctx.Logger.Debugf("updating all with default: %v\n", msg)
|
m.ctx.Logger.Debugf("updating all with default: %v\n", msg)
|
||||||
cmds = append(cmds, m.wm.UpdateAll(msg)...)
|
cmds = append(cmds, m.wm.UpdateAll(msg)...)
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
v, vcmd := m.views[m.currentView].Update(msg)
|
v, vcmd := m.views[m.currentView].Update(msg)
|
||||||
|
@ -15,16 +15,19 @@ import (
|
|||||||
"github.com/mrusme/gobbs/models/reply"
|
"github.com/mrusme/gobbs/models/reply"
|
||||||
"github.com/mrusme/gobbs/ui/cmd"
|
"github.com/mrusme/gobbs/ui/cmd"
|
||||||
"github.com/mrusme/gobbs/ui/ctx"
|
"github.com/mrusme/gobbs/ui/ctx"
|
||||||
|
"github.com/mrusme/gobbs/ui/windows/postdialog"
|
||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
VIEW_ID = "posts"
|
||||||
|
|
||||||
viewportStyle = lipgloss.NewStyle().
|
viewportStyle = lipgloss.NewStyle().
|
||||||
Margin(0, 0, 0, 0).
|
Margin(0, 0, 0, 0).
|
||||||
Padding(0, 0).
|
Padding(0, 0).
|
||||||
BorderTop(false).
|
BorderTop(false).
|
||||||
BorderLeft(false).
|
BorderLeft(false).
|
||||||
BorderRight(false).
|
BorderRight(false).
|
||||||
BorderBottom(false)
|
BorderBottom(false)
|
||||||
)
|
)
|
||||||
|
|
||||||
type KeyMap struct {
|
type KeyMap struct {
|
||||||
@ -145,7 +148,7 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
if ok {
|
if ok {
|
||||||
// m.ctx.Loading = true
|
// m.ctx.Loading = true
|
||||||
// cmds = append(cmds, m.loadItem(&i))
|
// cmds = append(cmds, m.loadItem(&i))
|
||||||
cmd := cmd.New(cmd.WinOpen, "post", cmd.Arg{
|
cmd := cmd.New(cmd.WinOpen, postdialog.WIN_ID, cmd.Arg{
|
||||||
Name: "post",
|
Name: "post",
|
||||||
Value: &i,
|
Value: &i,
|
||||||
})
|
})
|
||||||
@ -265,12 +268,6 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
// m.viewport.Height = viewportHeight + 1
|
// m.viewport.Height = viewportHeight + 1
|
||||||
// // cmds = append(cmds, viewport.Sync(m.viewport))
|
// // cmds = append(cmds, viewport.Sync(m.viewport))
|
||||||
|
|
||||||
case []list.Item:
|
|
||||||
m.items = msg
|
|
||||||
m.list.SetItems(m.items)
|
|
||||||
m.ctx.Loading = false
|
|
||||||
return m, nil
|
|
||||||
|
|
||||||
// case *post.Post:
|
// case *post.Post:
|
||||||
// m.viewport.SetContent(m.renderViewport(msg))
|
// m.viewport.SetContent(m.renderViewport(msg))
|
||||||
// m.WMOpen("post")
|
// m.WMOpen("post")
|
||||||
@ -280,20 +277,31 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
case cmd.Command:
|
case cmd.Command:
|
||||||
switch msg.Call {
|
switch msg.Call {
|
||||||
case cmd.ViewFocus:
|
case cmd.ViewFocus:
|
||||||
if msg.Target == "*" {
|
if msg.Target == VIEW_ID ||
|
||||||
|
msg.Target == "*" {
|
||||||
m.focused = true
|
m.focused = true
|
||||||
}
|
}
|
||||||
return m, nil
|
return m, nil
|
||||||
case cmd.ViewBlur:
|
case cmd.ViewBlur:
|
||||||
if msg.Target == "*" {
|
if msg.Target == VIEW_ID ||
|
||||||
|
msg.Target == "*" {
|
||||||
m.focused = false
|
m.focused = false
|
||||||
}
|
}
|
||||||
return m, nil
|
return m, nil
|
||||||
case cmd.ViewRefreshData:
|
case cmd.ViewRefreshData:
|
||||||
if msg.Target == "*" {
|
if msg.Target == VIEW_ID ||
|
||||||
|
msg.Target == "*" {
|
||||||
m.ctx.Loading = true
|
m.ctx.Loading = true
|
||||||
cmds = append(cmds, m.refresh())
|
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
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
@ -328,7 +336,13 @@ func (m *Model) refresh() tea.Cmd {
|
|||||||
items = append(items, post)
|
items = append(items, post)
|
||||||
}
|
}
|
||||||
|
|
||||||
return items
|
c := cmd.New(
|
||||||
|
cmd.ViewFreshData,
|
||||||
|
VIEW_ID,
|
||||||
|
cmd.Arg{Name: "items", Value: items},
|
||||||
|
)
|
||||||
|
|
||||||
|
return *c
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -23,7 +23,7 @@ func New() *WM {
|
|||||||
return wm
|
return wm
|
||||||
}
|
}
|
||||||
|
|
||||||
func (wm *WM) Open(id string, win windows.Window, xywh [4]int, args ...cmd.Arg) []tea.Cmd {
|
func (wm *WM) Open(id string, win windows.Window, xywh [4]int, command *cmd.Command) []tea.Cmd {
|
||||||
var tcmds []tea.Cmd
|
var tcmds []tea.Cmd
|
||||||
|
|
||||||
if wm.IsOpen(id) {
|
if wm.IsOpen(id) {
|
||||||
@ -40,18 +40,18 @@ func (wm *WM) Open(id string, win windows.Window, xywh [4]int, args ...cmd.Arg)
|
|||||||
|
|
||||||
wm.stack = append(wm.stack, *item)
|
wm.stack = append(wm.stack, *item)
|
||||||
|
|
||||||
tcmds = append(tcmds, wm.Update(id, tea.WindowSizeMsg{
|
// tcmds = append(tcmds, wm.Update(id, *command))
|
||||||
Width: item.XYWH[2],
|
// tcmds = append(tcmds, wm.Update(id, tea.WindowSizeMsg{
|
||||||
Height: item.XYWH[3],
|
// Width: item.XYWH[2],
|
||||||
}))
|
// Height: item.XYWH[3],
|
||||||
tcmds = append(tcmds, wm.Update(id, *cmd.New(
|
// }))
|
||||||
cmd.WinRefreshData,
|
// tcmds = append(tcmds, wm.Update(id, *cmd.New(
|
||||||
id,
|
// cmd.WinRefreshData,
|
||||||
args...,
|
// id,
|
||||||
)))
|
// )))
|
||||||
|
|
||||||
fcmds := wm.Focus(id)
|
// fcmds := wm.Focus(id)
|
||||||
tcmds = append(tcmds, fcmds...)
|
// tcmds = append(tcmds, fcmds...)
|
||||||
|
|
||||||
return tcmds
|
return tcmds
|
||||||
}
|
}
|
||||||
|
340
ui/windows/postcreatedialog/postcreatedialog.go
Normal file
340
ui/windows/postcreatedialog/postcreatedialog.go
Normal file
@ -0,0 +1,340 @@
|
|||||||
|
package postcreatedialog
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/charmbracelet/bubbles/key"
|
||||||
|
"github.com/charmbracelet/bubbles/textarea"
|
||||||
|
"github.com/charmbracelet/bubbles/viewport"
|
||||||
|
tea "github.com/charmbracelet/bubbletea"
|
||||||
|
"github.com/charmbracelet/glamour"
|
||||||
|
"github.com/charmbracelet/lipgloss"
|
||||||
|
"github.com/mrusme/gobbs/aggregator"
|
||||||
|
"github.com/mrusme/gobbs/models/post"
|
||||||
|
"github.com/mrusme/gobbs/models/reply"
|
||||||
|
"github.com/mrusme/gobbs/ui/cmd"
|
||||||
|
"github.com/mrusme/gobbs/ui/ctx"
|
||||||
|
"github.com/mrusme/gobbs/ui/helpers"
|
||||||
|
)
|
||||||
|
|
||||||
|
type KeyMap struct {
|
||||||
|
Refresh key.Binding
|
||||||
|
Select key.Binding
|
||||||
|
Esc key.Binding
|
||||||
|
Quit key.Binding
|
||||||
|
Reply key.Binding
|
||||||
|
}
|
||||||
|
|
||||||
|
var DefaultKeyMap = KeyMap{
|
||||||
|
Refresh: key.NewBinding(
|
||||||
|
key.WithKeys("ctrl+r"),
|
||||||
|
key.WithHelp("ctrl+r", "refresh"),
|
||||||
|
),
|
||||||
|
Select: key.NewBinding(
|
||||||
|
key.WithKeys("r", "enter"),
|
||||||
|
key.WithHelp("r/enter", "read"),
|
||||||
|
),
|
||||||
|
Esc: key.NewBinding(
|
||||||
|
key.WithKeys("esc"),
|
||||||
|
key.WithHelp("esc", "close"),
|
||||||
|
),
|
||||||
|
Quit: key.NewBinding(
|
||||||
|
key.WithKeys("q"),
|
||||||
|
),
|
||||||
|
Reply: key.NewBinding(
|
||||||
|
key.WithKeys("ctrl+s"),
|
||||||
|
key.WithHelp("ctrl+s", "reply"),
|
||||||
|
),
|
||||||
|
}
|
||||||
|
|
||||||
|
type Model struct {
|
||||||
|
ctx *ctx.Ctx
|
||||||
|
keymap KeyMap
|
||||||
|
textarea textarea.Model
|
||||||
|
focused bool
|
||||||
|
|
||||||
|
a *aggregator.Aggregator
|
||||||
|
glam *glamour.TermRenderer
|
||||||
|
|
||||||
|
activeReply *reply.Reply
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m Model) Init() tea.Cmd {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m Model) Focus() {
|
||||||
|
m.focused = true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m Model) Blur() {
|
||||||
|
m.focused = false
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewModel(c *ctx.Ctx) Model {
|
||||||
|
m := Model{
|
||||||
|
ctx: c,
|
||||||
|
keymap: DefaultKeyMap,
|
||||||
|
}
|
||||||
|
|
||||||
|
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.Reply):
|
||||||
|
replyToIdx, _ := strconv.Atoi(m.buffer)
|
||||||
|
|
||||||
|
m.ctx.Logger.Debugf("replyToIdx: %d", replyToIdx)
|
||||||
|
|
||||||
|
var irtID string = ""
|
||||||
|
var irtIRT string = ""
|
||||||
|
var irtSysIDX int = 0
|
||||||
|
|
||||||
|
if replyToIdx == 0 {
|
||||||
|
irtID = m.activePost.ID
|
||||||
|
irtSysIDX = m.activePost.SysIDX
|
||||||
|
} else {
|
||||||
|
irt := m.allReplies[(replyToIdx - 1)]
|
||||||
|
irtID = strconv.Itoa(replyToIdx + 1)
|
||||||
|
irtIRT = irt.InReplyTo
|
||||||
|
irtSysIDX = irt.SysIDX
|
||||||
|
}
|
||||||
|
|
||||||
|
r := reply.Reply{
|
||||||
|
ID: irtID,
|
||||||
|
InReplyTo: irtIRT,
|
||||||
|
Body: m.textarea.Value(),
|
||||||
|
SysIDX: irtSysIDX,
|
||||||
|
}
|
||||||
|
err := m.a.CreateReply(&r)
|
||||||
|
if err != nil {
|
||||||
|
m.ctx.Logger.Error(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
m.textarea.Reset()
|
||||||
|
m.buffer = ""
|
||||||
|
m.WMClose("reply")
|
||||||
|
return m, nil
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
case tea.WindowSizeMsg:
|
||||||
|
m.ctx.Logger.Debug("received WindowSizeMsg")
|
||||||
|
viewportWidth := m.ctx.Content[0] - 9
|
||||||
|
viewportHeight := m.ctx.Content[1] - 10
|
||||||
|
|
||||||
|
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 *post.Post:
|
||||||
|
m.ctx.Logger.Debug("got *post.Post")
|
||||||
|
m.activePost = msg
|
||||||
|
m.viewport.SetContent(m.renderViewport(m.activePost))
|
||||||
|
m.ctx.Loading = false
|
||||||
|
return m, nil
|
||||||
|
|
||||||
|
case cmd.Command:
|
||||||
|
m.ctx.Logger.Debugf("got command: %v\n", msg)
|
||||||
|
switch msg.Call {
|
||||||
|
case cmd.WinRefreshData:
|
||||||
|
if msg.Target == "post" {
|
||||||
|
m.activePost = msg.GetArg("post").(*post.Post)
|
||||||
|
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.WinFocus:
|
||||||
|
if msg.Target == "post" {
|
||||||
|
m.focused = true
|
||||||
|
}
|
||||||
|
return m, nil
|
||||||
|
case cmd.WinBlur:
|
||||||
|
if msg.Target == "post" {
|
||||||
|
m.focused = 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
|
||||||
|
|
||||||
|
m.viewport, cmd = m.viewport.Update(msg)
|
||||||
|
cmds = append(cmds, cmd)
|
||||||
|
|
||||||
|
return m, tea.Batch(cmds...)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Model) loadPost(p *post.Post) tea.Cmd {
|
||||||
|
return func() tea.Msg {
|
||||||
|
m.ctx.Logger.Debug("------ EXECUTED -----")
|
||||||
|
if err := m.a.LoadPost(p); err != nil {
|
||||||
|
m.ctx.Logger.Error(err)
|
||||||
|
}
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m Model) View() string {
|
||||||
|
return m.buildView(true)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m Model) buildView(cached bool) string {
|
||||||
|
var view strings.Builder = strings.Builder{}
|
||||||
|
|
||||||
|
var l string = ""
|
||||||
|
view.WriteString(lipgloss.JoinHorizontal(
|
||||||
|
lipgloss.Top,
|
||||||
|
l,
|
||||||
|
))
|
||||||
|
|
||||||
|
var style lipgloss.Style
|
||||||
|
if m.focused {
|
||||||
|
style = m.ctx.Theme.DialogBox.Titlebar.Focused
|
||||||
|
} else {
|
||||||
|
style = m.ctx.Theme.DialogBox.Titlebar.Blurred
|
||||||
|
}
|
||||||
|
titlebar := style.Align(lipgloss.Center).
|
||||||
|
Width(m.viewport.Width + 4).
|
||||||
|
Render("Post")
|
||||||
|
|
||||||
|
bottombar := m.ctx.Theme.DialogBox.Bottombar.
|
||||||
|
Width(m.viewport.Width + 4).
|
||||||
|
Render("[#]r reply · esc close")
|
||||||
|
|
||||||
|
ui := lipgloss.JoinVertical(
|
||||||
|
lipgloss.Center,
|
||||||
|
titlebar,
|
||||||
|
viewportStyle.Render(m.viewport.View()),
|
||||||
|
bottombar,
|
||||||
|
)
|
||||||
|
|
||||||
|
var tmp string
|
||||||
|
if m.focused {
|
||||||
|
tmp = helpers.PlaceOverlay(3, 2,
|
||||||
|
m.ctx.Theme.DialogBox.Window.Focused.Render(ui),
|
||||||
|
view.String(), true)
|
||||||
|
} else {
|
||||||
|
tmp = helpers.PlaceOverlay(3, 2,
|
||||||
|
m.ctx.Theme.DialogBox.Window.Blurred.Render(ui),
|
||||||
|
view.String(), true)
|
||||||
|
}
|
||||||
|
|
||||||
|
view = strings.Builder{}
|
||||||
|
view.WriteString(tmp)
|
||||||
|
|
||||||
|
return view.String()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Model) renderViewport(p *post.Post) string {
|
||||||
|
var out string = ""
|
||||||
|
|
||||||
|
var err error
|
||||||
|
m.glam, err = glamour.NewTermRenderer(
|
||||||
|
glamour.WithAutoStyle(),
|
||||||
|
glamour.WithWordWrap(m.viewport.Width),
|
||||||
|
)
|
||||||
|
if err != nil {
|
||||||
|
m.ctx.Logger.Error(err)
|
||||||
|
m.glam = nil
|
||||||
|
}
|
||||||
|
|
||||||
|
adj := "writes"
|
||||||
|
if p.Subject[len(p.Subject)-1:] == "?" {
|
||||||
|
adj = "asks"
|
||||||
|
}
|
||||||
|
|
||||||
|
body, err := m.glam.Render(p.Body)
|
||||||
|
if err != nil {
|
||||||
|
m.ctx.Logger.Error(err)
|
||||||
|
body = p.Body
|
||||||
|
}
|
||||||
|
out += fmt.Sprintf(
|
||||||
|
" %s\n\n %s\n%s",
|
||||||
|
m.ctx.Theme.Post.Author.Render(
|
||||||
|
fmt.Sprintf("%s %s:", p.Author.Name, adj),
|
||||||
|
),
|
||||||
|
m.ctx.Theme.Post.Subject.Render(p.Subject),
|
||||||
|
body,
|
||||||
|
)
|
||||||
|
|
||||||
|
m.replyIDs = []string{p.ID}
|
||||||
|
m.activePost = p
|
||||||
|
out += m.renderReplies(0, p.Author.Name, &p.Replies)
|
||||||
|
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
func (m *Model) renderReplies(
|
||||||
|
level int,
|
||||||
|
inReplyTo string,
|
||||||
|
replies *[]reply.Reply,
|
||||||
|
) string {
|
||||||
|
var out string = ""
|
||||||
|
|
||||||
|
if replies == nil {
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
for ri, re := range *replies {
|
||||||
|
var err error = nil
|
||||||
|
var body string = ""
|
||||||
|
var author string = ""
|
||||||
|
|
||||||
|
if re.Deleted {
|
||||||
|
body = "\n DELETED\n\n"
|
||||||
|
author = "DELETED"
|
||||||
|
} else {
|
||||||
|
body, err = m.glam.Render(re.Body)
|
||||||
|
if err != nil {
|
||||||
|
m.ctx.Logger.Error(err)
|
||||||
|
body = re.Body
|
||||||
|
}
|
||||||
|
|
||||||
|
author = re.Author.Name
|
||||||
|
}
|
||||||
|
|
||||||
|
m.replyIDs = append(m.replyIDs, re.ID)
|
||||||
|
m.allReplies = append(m.allReplies, &(*replies)[ri])
|
||||||
|
idx := len(m.replyIDs) - 1
|
||||||
|
|
||||||
|
out += fmt.Sprintf(
|
||||||
|
"\n\n %s %s%s%s\n%s",
|
||||||
|
m.ctx.Theme.Reply.Author.Render(
|
||||||
|
author,
|
||||||
|
),
|
||||||
|
lipgloss.NewStyle().
|
||||||
|
Foreground(m.ctx.Theme.Reply.Author.GetBackground()).
|
||||||
|
Render(fmt.Sprintf("writes in reply to %s:", inReplyTo)),
|
||||||
|
strings.Repeat(" ", (m.viewport.Width-len(author)-len(inReplyTo)-28)),
|
||||||
|
lipgloss.NewStyle().
|
||||||
|
Foreground(lipgloss.Color("#777777")).
|
||||||
|
Render(fmt.Sprintf("#%d", idx)),
|
||||||
|
body,
|
||||||
|
)
|
||||||
|
|
||||||
|
idx++
|
||||||
|
out += m.renderReplies(level+1, re.Author.Name, &re.Replies)
|
||||||
|
}
|
||||||
|
|
||||||
|
return out
|
||||||
|
}
|
@ -19,13 +19,15 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
var (
|
var (
|
||||||
|
WIN_ID = "postShow"
|
||||||
|
|
||||||
viewportStyle = lipgloss.NewStyle().
|
viewportStyle = lipgloss.NewStyle().
|
||||||
Margin(0, 0, 0, 0).
|
Margin(0, 0, 0, 0).
|
||||||
Padding(0, 0).
|
Padding(0, 0).
|
||||||
BorderTop(false).
|
BorderTop(false).
|
||||||
BorderLeft(false).
|
BorderLeft(false).
|
||||||
BorderRight(false).
|
BorderRight(false).
|
||||||
BorderBottom(false)
|
BorderBottom(false)
|
||||||
)
|
)
|
||||||
|
|
||||||
type KeyMap struct {
|
type KeyMap struct {
|
||||||
@ -153,34 +155,40 @@ func (m Model) Update(msg tea.Msg) (tea.Model, tea.Cmd) {
|
|||||||
m.viewport.Height = viewportHeight + 1
|
m.viewport.Height = viewportHeight + 1
|
||||||
// cmds = append(cmds, viewport.Sync(m.viewport))
|
// cmds = append(cmds, viewport.Sync(m.viewport))
|
||||||
|
|
||||||
case *post.Post:
|
|
||||||
m.ctx.Logger.Debug("got *post.Post")
|
|
||||||
m.activePost = msg
|
|
||||||
m.viewport.SetContent(m.renderViewport(m.activePost))
|
|
||||||
m.ctx.Loading = false
|
|
||||||
return m, nil
|
|
||||||
|
|
||||||
case cmd.Command:
|
case cmd.Command:
|
||||||
m.ctx.Logger.Debugf("got command: %v\n", msg)
|
m.ctx.Logger.Debugf("got command: %v\n", msg)
|
||||||
switch msg.Call {
|
switch msg.Call {
|
||||||
case cmd.WinRefreshData:
|
case cmd.WinOpen, cmd.WinRefreshData:
|
||||||
if msg.Target == "post" {
|
if msg.Target == WIN_ID {
|
||||||
|
m.ctx.Logger.Debug("got own WinOpen command")
|
||||||
m.activePost = msg.GetArg("post").(*post.Post)
|
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.Logger.Debugf("loading post: %v", m.activePost.ID)
|
||||||
m.ctx.Loading = true
|
m.ctx.Loading = true
|
||||||
return m, m.loadPost(m.activePost)
|
return m, m.loadPost(m.activePost)
|
||||||
}
|
}
|
||||||
return m, nil
|
return m, nil
|
||||||
case cmd.WinFocus:
|
case cmd.WinFocus:
|
||||||
if msg.Target == "post" {
|
if msg.Target == WIN_ID ||
|
||||||
|
msg.Target == "*" {
|
||||||
m.focused = true
|
m.focused = true
|
||||||
}
|
}
|
||||||
return m, nil
|
return m, nil
|
||||||
case cmd.WinBlur:
|
case cmd.WinBlur:
|
||||||
if msg.Target == "post" {
|
if msg.Target == WIN_ID ||
|
||||||
|
msg.Target == "*" {
|
||||||
m.focused = false
|
m.focused = false
|
||||||
}
|
}
|
||||||
return m, nil
|
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:
|
default:
|
||||||
m.ctx.Logger.Debugf("received unhandled command: %v\n", msg)
|
m.ctx.Logger.Debugf("received unhandled command: %v\n", msg)
|
||||||
}
|
}
|
||||||
@ -203,7 +211,13 @@ func (m *Model) loadPost(p *post.Post) tea.Cmd {
|
|||||||
if err := m.a.LoadPost(p); err != nil {
|
if err := m.a.LoadPost(p); err != nil {
|
||||||
m.ctx.Logger.Error(err)
|
m.ctx.Logger.Error(err)
|
||||||
}
|
}
|
||||||
return p
|
|
||||||
|
c := cmd.New(
|
||||||
|
cmd.WinFreshData,
|
||||||
|
WIN_ID,
|
||||||
|
cmd.Arg{Name: "post", Value: p},
|
||||||
|
)
|
||||||
|
return *c
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user