state.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118
  1. package cmd
  2. import (
  3. "context"
  4. "fmt"
  5. "log/slog"
  6. "runtime"
  7. "github.com/ayn2op/discordo/internal/notifications"
  8. "github.com/diamondburned/arikawa/v3/api"
  9. "github.com/diamondburned/arikawa/v3/gateway"
  10. "github.com/diamondburned/arikawa/v3/utils/httputil/httpdriver"
  11. "github.com/diamondburned/ningen/v3"
  12. "github.com/gdamore/tcell/v2"
  13. "github.com/rivo/tview"
  14. )
  15. type State struct {
  16. *ningen.State
  17. }
  18. func openState(token string) error {
  19. api.UserAgent = app.cfg.Identify.UserAgent
  20. gateway.DefaultIdentity = gateway.IdentifyProperties{
  21. OS: runtime.GOOS,
  22. Device: "",
  23. Browser: app.cfg.Identify.Browser,
  24. BrowserVersion: app.cfg.Identify.BrowserVersion,
  25. BrowserUserAgent: app.cfg.Identify.UserAgent,
  26. }
  27. gateway.DefaultPresence = &gateway.UpdatePresenceCommand{
  28. Status: app.cfg.Identify.Status,
  29. }
  30. discordState = &State{
  31. State: ningen.New(token),
  32. }
  33. // Handlers
  34. discordState.AddHandler(discordState.onEvent)
  35. discordState.AddHandler(discordState.onReady)
  36. discordState.AddHandler(discordState.onMessageCreate)
  37. discordState.AddHandler(discordState.onMessageDelete)
  38. discordState.AddHandler(discordState.onGetMemberChunk)
  39. discordState.StateLog = discordState.onStateLog
  40. discordState.OnRequest = append(discordState.OnRequest, discordState.onRequest)
  41. return discordState.Open(context.TODO())
  42. }
  43. func (s *State) onStateLog(err error) {
  44. slog.Error("state log", "err", err)
  45. }
  46. func (s *State) onRequest(r httpdriver.Request) error {
  47. req, ok := r.(*httpdriver.DefaultRequest)
  48. if ok {
  49. slog.Debug("new HTTP request", "method", req.Method, "url", req.URL)
  50. }
  51. return nil
  52. }
  53. func (s *State) onEvent(event any) {
  54. slog.Debug("new gateway event", "name", fmt.Sprintf("%T", event), "data", event)
  55. }
  56. func (s *State) onReady(r *gateway.ReadyEvent) {
  57. root := app.guildsTree.GetRoot()
  58. root.ClearChildren()
  59. dmNode := tview.NewTreeNode("Direct Messages")
  60. dmNode.SetColor(tcell.GetColor(app.cfg.Theme.GuildsTree.PrivateChannelColor))
  61. root.AddChild(dmNode)
  62. for _, folder := range r.UserSettings.GuildFolders {
  63. if folder.ID == 0 && len(folder.GuildIDs) == 1 {
  64. g, err := discordState.Cabinet.Guild(folder.GuildIDs[0])
  65. if err != nil {
  66. slog.Error("failed to get guild from state", "guild_id", folder.GuildIDs[0], "err", err)
  67. continue
  68. }
  69. app.guildsTree.createGuildNode(root, *g)
  70. } else {
  71. app.guildsTree.createFolderNode(folder)
  72. }
  73. }
  74. app.guildsTree.SetCurrentNode(root)
  75. app.SetFocus(app.guildsTree)
  76. }
  77. func (s *State) onMessageCreate(m *gateway.MessageCreateEvent) {
  78. if app.guildsTree.selectedChannelID.IsValid() && app.guildsTree.selectedChannelID == m.ChannelID {
  79. app.messagesText.createMessage(m.Message)
  80. }
  81. if err := notifications.HandleIncomingMessage(*s.State, m, app.cfg); err != nil {
  82. slog.Error("Notification failed", "err", err)
  83. }
  84. }
  85. func (s *State) onMessageDelete(m *gateway.MessageDeleteEvent) {
  86. if app.guildsTree.selectedChannelID == m.ChannelID {
  87. app.messagesText.selectedMessageID = 0
  88. app.messagesText.Highlight()
  89. app.messagesText.Clear()
  90. app.messagesText.drawMsgs(m.ChannelID)
  91. }
  92. }
  93. func (s *State) onGetMemberChunk(g *gateway.GuildMembersChunkEvent) {
  94. app.messagesText.setFetchingChunk(false)
  95. }