state.go 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177
  1. package cmd
  2. import (
  3. "context"
  4. "log/slog"
  5. "github.com/ayn2op/discordo/internal/http"
  6. "github.com/ayn2op/discordo/internal/notifications"
  7. "github.com/ayn2op/tview"
  8. "github.com/diamondburned/arikawa/v3/gateway"
  9. "github.com/diamondburned/arikawa/v3/session"
  10. "github.com/diamondburned/arikawa/v3/state"
  11. "github.com/diamondburned/arikawa/v3/state/store/defaultstore"
  12. "github.com/diamondburned/arikawa/v3/utils/handler"
  13. "github.com/diamondburned/arikawa/v3/utils/httputil"
  14. "github.com/diamondburned/arikawa/v3/utils/httputil/httpdriver"
  15. "github.com/diamondburned/arikawa/v3/utils/ws"
  16. "github.com/diamondburned/ningen/v3"
  17. "github.com/diamondburned/ningen/v3/states/read"
  18. )
  19. func openState(token string) error {
  20. identifyProps := http.IdentifyProperties()
  21. gateway.DefaultIdentity = identifyProps
  22. gateway.DefaultPresence = &gateway.UpdatePresenceCommand{
  23. Status: app.cfg.Status,
  24. }
  25. id := gateway.DefaultIdentifier(token)
  26. id.Compress = false
  27. session := session.NewCustom(id, http.NewClient(token), handler.New())
  28. state := state.NewFromSession(session, defaultstore.New())
  29. discordState = ningen.FromState(state)
  30. // Handlers
  31. discordState.AddHandler(onRaw)
  32. discordState.AddHandler(onReady)
  33. discordState.AddHandler(onMessageCreate)
  34. discordState.AddHandler(onMessageUpdate)
  35. discordState.AddHandler(onMessageDelete)
  36. discordState.AddHandler(onReadUpdate)
  37. discordState.AddHandler(func(event *gateway.GuildMembersChunkEvent) {
  38. app.chatView.messagesList.setFetchingChunk(false, uint(len(event.Members)))
  39. })
  40. discordState.AddHandler(func(event *gateway.GuildMemberRemoveEvent) {
  41. app.chatView.messageInput.cache.Invalidate(event.GuildID.String()+" "+event.User.Username, discordState.MemberState.SearchLimit)
  42. })
  43. discordState.StateLog = func(err error) {
  44. slog.Error("state log", "err", err)
  45. }
  46. discordState.OnRequest = append(discordState.OnRequest, httputil.WithHeaders(http.Headers()), onRequest)
  47. return discordState.Open(context.TODO())
  48. }
  49. func onRequest(r httpdriver.Request) error {
  50. if req, ok := r.(*httpdriver.DefaultRequest); ok {
  51. slog.Debug("new HTTP request", "method", req.Method, "url", req.URL)
  52. }
  53. return nil
  54. }
  55. func onRaw(event *ws.RawEvent) {
  56. slog.Debug(
  57. "new raw event",
  58. "code", event.OriginalCode,
  59. "type", event.OriginalType,
  60. // "data", event.Raw,
  61. )
  62. }
  63. func onReadUpdate(event *read.UpdateEvent) {
  64. var guildNode *tview.TreeNode
  65. app.chatView.guildsTree.
  66. GetRoot().
  67. Walk(func(node, parent *tview.TreeNode) bool {
  68. switch node.GetReference() {
  69. case event.GuildID:
  70. node.SetTextStyle(app.chatView.guildsTree.getGuildNodeStyle(event.GuildID))
  71. guildNode = node
  72. return false
  73. case event.ChannelID:
  74. // private channel
  75. if !event.GuildID.IsValid() {
  76. style := app.chatView.guildsTree.getChannelNodeStyle(event.ChannelID)
  77. node.SetTextStyle(style)
  78. return false
  79. }
  80. }
  81. return true
  82. })
  83. if guildNode != nil {
  84. guildNode.Walk(func(node, parent *tview.TreeNode) bool {
  85. if node.GetReference() == event.ChannelID {
  86. node.SetTextStyle(app.chatView.guildsTree.getChannelNodeStyle(event.ChannelID))
  87. return false
  88. }
  89. return true
  90. })
  91. }
  92. app.Draw()
  93. }
  94. func onReady(r *gateway.ReadyEvent) {
  95. dmNode := tview.NewTreeNode("Direct Messages")
  96. root := app.chatView.guildsTree.
  97. GetRoot().
  98. ClearChildren().
  99. AddChild(dmNode)
  100. for _, folder := range r.UserSettings.GuildFolders {
  101. if folder.ID == 0 && len(folder.GuildIDs) == 1 {
  102. guild, err := discordState.Cabinet.Guild(folder.GuildIDs[0])
  103. if err != nil {
  104. slog.Error(
  105. "failed to get guild from state",
  106. "guild_id",
  107. folder.GuildIDs[0],
  108. "err",
  109. err,
  110. )
  111. continue
  112. }
  113. app.chatView.guildsTree.createGuildNode(root, *guild)
  114. } else {
  115. app.chatView.guildsTree.createFolderNode(folder)
  116. }
  117. }
  118. app.chatView.guildsTree.SetCurrentNode(root)
  119. app.SetFocus(app.chatView.guildsTree)
  120. app.Draw()
  121. }
  122. func onMessageCreate(message *gateway.MessageCreateEvent) {
  123. if app.chatView.selectedChannel != nil &&
  124. app.chatView.selectedChannel.ID == message.ChannelID {
  125. app.chatView.messagesList.drawMessage(app.chatView.messagesList, message.Message)
  126. app.Draw()
  127. }
  128. if err := notifications.Notify(discordState, message, app.cfg); err != nil {
  129. slog.Error("Notification failed", "err", err)
  130. }
  131. }
  132. func onMessageUpdate(message *gateway.MessageUpdateEvent) {
  133. if app.chatView.selectedChannel != nil &&
  134. app.chatView.selectedChannel.ID == message.ChannelID {
  135. onMessageDelete(&gateway.MessageDeleteEvent{ID: message.ID, ChannelID: message.ChannelID, GuildID: message.GuildID})
  136. }
  137. }
  138. func onMessageDelete(message *gateway.MessageDeleteEvent) {
  139. if app.chatView.selectedChannel != nil &&
  140. app.chatView.selectedChannel.ID == message.ChannelID {
  141. messages, err := discordState.Cabinet.Messages(message.ChannelID)
  142. if err != nil {
  143. slog.Error("failed to get messages from state", "err", err, "channel_id", message.ChannelID)
  144. return
  145. }
  146. app.chatView.messagesList.reset()
  147. app.chatView.messagesList.drawMessages(messages)
  148. app.Draw()
  149. }
  150. }