state.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. package cmd
  2. import (
  3. "context"
  4. "log/slog"
  5. "runtime"
  6. "github.com/ayn2op/discordo/internal/notifications"
  7. "github.com/ayn2op/tview"
  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/arikawa/v3/utils/ws"
  12. "github.com/diamondburned/ningen/v3"
  13. "github.com/gdamore/tcell/v2"
  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.onReady)
  35. discordState.AddHandler(discordState.onMessageCreate)
  36. discordState.AddHandler(discordState.onMessageDelete)
  37. discordState.AddHandler(func(_ *gateway.GuildMembersChunkEvent) {
  38. app.messagesText.setFetchingChunk(false)
  39. })
  40. discordState.AddHandler(func(event *ws.RawEvent) {
  41. slog.Debug(
  42. "new raw event",
  43. "code",
  44. event.OriginalCode,
  45. "type",
  46. event.OriginalType,
  47. "data",
  48. event.Raw,
  49. )
  50. })
  51. discordState.StateLog = func(err error) {
  52. slog.Error("state log", "err", err)
  53. }
  54. discordState.OnRequest = append(discordState.OnRequest, discordState.onRequest)
  55. return discordState.Open(context.TODO())
  56. }
  57. func (s *state) onRequest(r httpdriver.Request) error {
  58. req, ok := r.(*httpdriver.DefaultRequest)
  59. if ok {
  60. slog.Debug("new HTTP request", "method", req.Method, "url", req.URL)
  61. }
  62. return nil
  63. }
  64. func (s *state) onReady(ready *gateway.ReadyEvent) {
  65. root := app.guildsTree.GetRoot()
  66. root.ClearChildren()
  67. dmNode := tview.NewTreeNode("Direct Messages")
  68. dmNode.SetColor(tcell.GetColor(app.cfg.Theme.GuildsTree.PrivateChannelColor))
  69. root.AddChild(dmNode)
  70. for _, folder := range ready.UserSettings.GuildFolders {
  71. if folder.ID == 0 && len(folder.GuildIDs) == 1 {
  72. guild, err := discordState.Cabinet.Guild(folder.GuildIDs[0])
  73. if err != nil {
  74. slog.Error(
  75. "failed to get guild from state",
  76. "guild_id",
  77. folder.GuildIDs[0],
  78. "err",
  79. err,
  80. )
  81. continue
  82. }
  83. app.guildsTree.createGuildNode(root, *guild)
  84. } else {
  85. app.guildsTree.createFolderNode(folder)
  86. }
  87. }
  88. app.guildsTree.SetCurrentNode(root)
  89. app.SetFocus(app.guildsTree)
  90. }
  91. func (s *state) onMessageCreate(msg *gateway.MessageCreateEvent) {
  92. if app.guildsTree.selectedChannelID.IsValid() &&
  93. app.guildsTree.selectedChannelID == msg.ChannelID {
  94. app.messagesText.createMsg(msg.Message)
  95. }
  96. if err := notifications.HandleIncomingMessage(s.State, msg, app.cfg); err != nil {
  97. slog.Error("Notification failed", "err", err)
  98. }
  99. }
  100. func (s *state) onMessageDelete(msg *gateway.MessageDeleteEvent) {
  101. if app.guildsTree.selectedChannelID == msg.ChannelID {
  102. app.messagesText.selectedMessageID = 0
  103. app.messagesText.Highlight()
  104. app.messagesText.Clear()
  105. app.messagesText.drawMsgs(msg.ChannelID)
  106. }
  107. }