state.go 3.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129
  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. func openState(token string) error {
  16. api.UserAgent = app.cfg.Identify.UserAgent
  17. gateway.DefaultIdentity = gateway.IdentifyProperties{
  18. OS: runtime.GOOS,
  19. Device: "",
  20. Browser: app.cfg.Identify.Browser,
  21. BrowserVersion: app.cfg.Identify.BrowserVersion,
  22. BrowserUserAgent: app.cfg.Identify.UserAgent,
  23. }
  24. gateway.DefaultPresence = &gateway.UpdatePresenceCommand{
  25. Status: app.cfg.Identify.Status,
  26. }
  27. discordState = ningen.New(token)
  28. // Handlers
  29. discordState.AddHandler(onReady)
  30. discordState.AddHandler(onMessageCreate)
  31. discordState.AddHandler(onMessageDelete)
  32. discordState.AddHandler(func(event *gateway.GuildMembersChunkEvent) {
  33. app.messagesText.setFetchingChunk(false, uint(len(event.Members)))
  34. })
  35. discordState.AddHandler(func(event *gateway.GuildMemberRemoveEvent) {
  36. app.messageInput.cache.Invalidate(event.GuildID.String()+" "+event.User.Username, discordState.MemberState.SearchLimit)
  37. })
  38. discordState.AddHandler(func(event *ws.RawEvent) {
  39. slog.Debug(
  40. "new raw event",
  41. "code",
  42. event.OriginalCode,
  43. "type",
  44. event.OriginalType,
  45. "data",
  46. event.Raw,
  47. )
  48. })
  49. discordState.StateLog = func(err error) {
  50. slog.Error("state log", "err", err)
  51. }
  52. discordState.OnRequest = append(discordState.OnRequest, onRequest)
  53. return discordState.Open(context.TODO())
  54. }
  55. func onRequest(r httpdriver.Request) error {
  56. req, ok := r.(*httpdriver.DefaultRequest)
  57. if ok {
  58. slog.Debug("new HTTP request", "method", req.Method, "url", req.URL)
  59. }
  60. return nil
  61. }
  62. func onReady(r *gateway.ReadyEvent) {
  63. root := app.guildsTree.GetRoot()
  64. root.ClearChildren()
  65. dmNode := tview.NewTreeNode("Direct Messages").SetColor(tcell.GetColor(app.cfg.Theme.GuildsTree.PrivateChannelColor))
  66. root.AddChild(dmNode)
  67. for _, folder := range r.UserSettings.GuildFolders {
  68. if folder.ID == 0 && len(folder.GuildIDs) == 1 {
  69. guild, err := discordState.Cabinet.Guild(folder.GuildIDs[0])
  70. if err != nil {
  71. slog.Error(
  72. "failed to get guild from state",
  73. "guild_id",
  74. folder.GuildIDs[0],
  75. "err",
  76. err,
  77. )
  78. continue
  79. }
  80. app.guildsTree.createGuildNode(root, *guild)
  81. } else {
  82. app.guildsTree.createFolderNode(folder)
  83. }
  84. }
  85. app.guildsTree.SetCurrentNode(root)
  86. app.SetFocus(app.guildsTree)
  87. }
  88. func onMessageCreate(m *gateway.MessageCreateEvent) {
  89. if app.guildsTree.selectedChannelID.IsValid() &&
  90. app.guildsTree.selectedChannelID == m.ChannelID {
  91. app.messagesText.createMsg(m.Message)
  92. app.Draw()
  93. }
  94. if err := notifications.HandleIncomingMessage(discordState, m, app.cfg); err != nil {
  95. slog.Error("Notification failed", "err", err)
  96. }
  97. }
  98. func onMessageDelete(m *gateway.MessageDeleteEvent) {
  99. if app.guildsTree.selectedChannelID == m.ChannelID {
  100. app.messagesText.selectedMessageID = 0
  101. app.messagesText.Highlight()
  102. app.messagesText.Clear()
  103. app.messagesText.drawMsgs(m.ChannelID)
  104. }
  105. }