| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180 |
- package ui
- import (
- "log"
- "github.com/ayn2op/discordo/discordmd"
- "github.com/diamondburned/arikawa/v3/discord"
- "github.com/gdamore/tcell/v2"
- "github.com/rivo/tview"
- )
- type MessagesView struct {
- *tview.TextView
- // The index of the currently selected message. A negative index indicates that there is no currently selected message.
- selected int
- app *Application
- }
- func newMessagesView(app *Application) *MessagesView {
- v := &MessagesView{
- TextView: tview.NewTextView(),
- selected: -1,
- app: app,
- }
- v.SetDynamicColors(true)
- v.SetRegions(true)
- v.SetWordWrap(true)
- v.SetInputCapture(v.onInputCapture)
- v.SetChangedFunc(func() {
- v.app.Draw()
- })
- v.SetTitle("Messages")
- v.SetTitleAlign(tview.AlignLeft)
- v.SetBorder(true)
- v.SetBorderPadding(0, 0, 1, 1)
- return v
- }
- func (v *MessagesView) setTitle(c *discord.Channel) {
- title := channelToString(*c)
- if c.Topic != "" {
- title += " - " + discordmd.Parse(c.Topic)
- }
- v.SetTitle(title)
- }
- func (v *MessagesView) loadMessages(c *discord.Channel) {
- // The returned slice will be sorted from latest to oldest.
- ms, err := v.app.state.Messages(c.ID, v.app.config.MessagesLimit)
- if err != nil {
- log.Println(err)
- return
- }
- for i := len(ms) - 1; i >= 0; i-- {
- _, err = v.app.view.MessagesView.Write(buildMessage(v.app, ms[i]))
- if err != nil {
- log.Println(err)
- continue
- }
- }
- v.ScrollToEnd()
- }
- func (v *MessagesView) onInputCapture(e *tcell.EventKey) *tcell.EventKey {
- if v.app.view.ChannelsView.selected == nil {
- return nil
- }
- // Messages should return messages ordered from latest to earliest.
- ms, err := v.app.state.Cabinet.Messages(v.app.view.ChannelsView.selected.ID)
- if err != nil || len(ms) == 0 {
- return nil
- }
- switch e.Name() {
- case v.app.config.Keys.MessagesView.OpenActionsView:
- return v.openActionsView(ms)
- case v.app.config.Keys.MessagesView.SelectPreviousMessage:
- return v.selectPreviousMessage(ms)
- case v.app.config.Keys.MessagesView.SelectNextMessage:
- return v.selectNextMessage(ms)
- case v.app.config.Keys.MessagesView.SelectFirstMessage:
- return v.selectFirstMessage(ms)
- case v.app.config.Keys.MessagesView.SelectLastMessage:
- return v.selectLastMessage(ms)
- case "Esc":
- v.selected = -1
- v.app.SetFocus(v.app.view)
- v.
- Clear().
- Highlight().
- SetTitle("")
- return nil
- }
- return e
- }
- func (v *MessagesView) selectPreviousMessage(ms []discord.Message) *tcell.EventKey {
- // If there are no highlighted regions, select the latest (last) message.
- if len(v.GetHighlights()) == 0 {
- v.selected = 0
- } else {
- // If the selected message is the oldest (first) message, select the latest (last) message.
- if v.selected == len(ms)-1 {
- v.selected = 0
- } else {
- v.selected++
- }
- }
- v.Highlight(ms[v.selected].ID.String())
- v.ScrollToHighlight()
- return nil
- }
- func (v *MessagesView) selectNextMessage(ms []discord.Message) *tcell.EventKey {
- // If there are no highlighted regions, select the latest (last) message.
- if len(v.GetHighlights()) == 0 {
- v.selected = 0
- } else {
- // If the selected message is the latest (last) message, select the oldest (first) message.
- if v.selected == 0 {
- v.selected = len(ms) - 1
- } else {
- v.selected--
- }
- }
- v.
- Highlight(ms[v.selected].ID.String()).
- ScrollToHighlight()
- return nil
- }
- func (v *MessagesView) selectFirstMessage(ms []discord.Message) *tcell.EventKey {
- v.selected = len(ms) - 1
- v.
- Highlight(ms[v.selected].ID.String()).
- ScrollToHighlight()
- return nil
- }
- func (v *MessagesView) selectLastMessage(ms []discord.Message) *tcell.EventKey {
- v.selected = 0
- v.
- Highlight(ms[v.selected].ID.String()).
- ScrollToHighlight()
- return nil
- }
- func (v *MessagesView) openActionsView(ms []discord.Message) *tcell.EventKey {
- hs := v.GetHighlights()
- if len(hs) == 0 {
- return nil
- }
- mID, err := discord.ParseSnowflake(hs[0])
- if err != nil {
- return nil
- }
- _, m := findMessageByID(ms, discord.MessageID(mID))
- if m == nil {
- return nil
- }
- actionsView := newActionsView(v.app, m)
- v.app.SetRoot(actionsView, true)
- return nil
- }
|