state.go 5.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211
  1. package chat
  2. import (
  3. "context"
  4. "log/slog"
  5. "slices"
  6. "github.com/ayn2op/discordo/internal/http"
  7. "github.com/ayn2op/discordo/internal/notifications"
  8. "github.com/ayn2op/tview"
  9. "github.com/diamondburned/arikawa/v3/discord"
  10. "github.com/diamondburned/arikawa/v3/gateway"
  11. "github.com/diamondburned/arikawa/v3/session"
  12. "github.com/diamondburned/arikawa/v3/state"
  13. "github.com/diamondburned/arikawa/v3/state/store/defaultstore"
  14. "github.com/diamondburned/arikawa/v3/utils/handler"
  15. "github.com/diamondburned/arikawa/v3/utils/httputil"
  16. "github.com/diamondburned/arikawa/v3/utils/httputil/httpdriver"
  17. "github.com/diamondburned/arikawa/v3/utils/ws"
  18. "github.com/diamondburned/ningen/v3"
  19. )
  20. func (v *View) OpenState(token string) error {
  21. identifyProps := http.IdentifyProperties()
  22. gateway.DefaultIdentity = identifyProps
  23. gateway.DefaultPresence = &gateway.UpdatePresenceCommand{
  24. Status: v.cfg.Status,
  25. }
  26. id := gateway.DefaultIdentifier(token)
  27. id.Compress = false
  28. session := session.NewCustom(id, http.NewClient(token), handler.New())
  29. state := state.NewFromSession(session, defaultstore.New())
  30. v.state = ningen.FromState(state)
  31. // Handlers
  32. v.state.AddHandler(v.onRaw)
  33. v.state.AddHandler(v.onReady)
  34. v.state.AddHandler(v.onMessageCreate)
  35. v.state.AddHandler(v.onMessageUpdate)
  36. v.state.AddHandler(v.onMessageDelete)
  37. v.state.AddHandler(v.onReadUpdate)
  38. v.state.AddHandler(v.onGuildMembersChunk)
  39. v.state.AddHandler(v.onGuildMemberRemove)
  40. if v.cfg.TypingIndicator.Receive {
  41. v.state.AddHandler(v.onTypingStart)
  42. }
  43. v.state.StateLog = func(err error) {
  44. slog.Error("state log", "err", err)
  45. }
  46. v.state.OnRequest = append(v.state.OnRequest, httputil.WithHeaders(http.Headers()), v.onRequest)
  47. return v.state.Open(context.TODO())
  48. }
  49. func (v *View) CloseState() error {
  50. if v.state == nil {
  51. return nil
  52. }
  53. return v.state.Close()
  54. }
  55. func (v *View) onRequest(r httpdriver.Request) error {
  56. if req, ok := r.(*httpdriver.DefaultRequest); ok {
  57. slog.Debug("new HTTP request", "method", req.Method, "url", req.URL)
  58. }
  59. return nil
  60. }
  61. func (v *View) onRaw(event *ws.RawEvent) {
  62. slog.Debug(
  63. "new raw event",
  64. "code", event.OriginalCode,
  65. "type", event.OriginalType,
  66. // "data", event.Raw,
  67. )
  68. }
  69. func (v *View) onReady(r *gateway.ReadyEvent) {
  70. dmNode := tview.NewTreeNode("Direct Messages").SetReference(dmNode{})
  71. // Rebuild indexes from scratch so reconnects and account switches do not
  72. // retain stale pointers to detached tree nodes.
  73. v.guildsTree.resetNodeIndex()
  74. v.guildsTree.dmRootNode = dmNode
  75. root := v.guildsTree.
  76. GetRoot().
  77. ClearChildren().
  78. AddChild(dmNode)
  79. for _, folder := range r.UserSettings.GuildFolders {
  80. if folder.ID == 0 && len(folder.GuildIDs) == 1 {
  81. guild, err := v.state.Cabinet.Guild(folder.GuildIDs[0])
  82. if err != nil {
  83. slog.Error(
  84. "failed to get guild from state",
  85. "guild_id",
  86. folder.GuildIDs[0],
  87. "err",
  88. err,
  89. )
  90. continue
  91. }
  92. v.guildsTree.createGuildNode(root, *guild)
  93. } else {
  94. v.guildsTree.createFolderNode(folder)
  95. }
  96. }
  97. v.guildsTree.SetCurrentNode(root)
  98. v.app.SetFocus(v.guildsTree)
  99. v.app.Draw()
  100. }
  101. func (v *View) onMessageCreate(message *gateway.MessageCreateEvent) {
  102. selectedChannel := v.SelectedChannel()
  103. if selectedChannel != nil && selectedChannel.ID == message.ChannelID {
  104. v.removeTyper(message.Author.ID)
  105. v.app.QueueUpdateDraw(func() {
  106. v.messagesList.addMessage(message.Message)
  107. })
  108. } else {
  109. if err := notifications.Notify(v.state, message, v.cfg); err != nil {
  110. slog.Error("failed to notify", "err", err, "channel_id", message.ChannelID, "message_id", message.ID)
  111. }
  112. }
  113. }
  114. func (v *View) onMessageUpdate(message *gateway.MessageUpdateEvent) {
  115. if selected := v.SelectedChannel(); selected != nil && selected.ID == message.ChannelID {
  116. index := slices.IndexFunc(v.messagesList.messages, func(m discord.Message) bool {
  117. return m.ID == message.ID
  118. })
  119. if index < 0 {
  120. return
  121. }
  122. v.app.QueueUpdateDraw(func() {
  123. v.messagesList.setMessage(index, message.Message)
  124. })
  125. }
  126. }
  127. func (v *View) onMessageDelete(message *gateway.MessageDeleteEvent) {
  128. if selected := v.SelectedChannel(); selected != nil && selected.ID == message.ChannelID {
  129. prevCursor := v.messagesList.Cursor()
  130. deletedIndex := slices.IndexFunc(v.messagesList.messages, func(m discord.Message) bool {
  131. return m.ID == message.ID
  132. })
  133. if deletedIndex < 0 {
  134. return
  135. }
  136. v.app.QueueUpdateDraw(func() {
  137. v.messagesList.deleteMessage(deletedIndex)
  138. })
  139. // Keep cursor stable when possible after removal.
  140. newCursor := prevCursor
  141. if prevCursor == deletedIndex {
  142. // Prefer previous item; fall forward if we deleted the first.
  143. newCursor = deletedIndex - 1
  144. if newCursor < 0 {
  145. if deletedIndex < len(v.messagesList.messages) {
  146. newCursor = deletedIndex
  147. } else {
  148. newCursor = -1
  149. }
  150. }
  151. } else if prevCursor > deletedIndex {
  152. // Shift back since the list shrank before the cursor.
  153. newCursor = prevCursor - 1
  154. }
  155. if newCursor != prevCursor {
  156. // Avoid redundant cursor updates if nothing changed.
  157. v.app.QueueUpdateDraw(func() {
  158. v.messagesList.SetCursor(newCursor)
  159. })
  160. }
  161. }
  162. }
  163. func (v *View) onGuildMembersChunk(event *gateway.GuildMembersChunkEvent) {
  164. v.messagesList.setFetchingChunk(false, uint(len(event.Members)))
  165. }
  166. func (v *View) onGuildMemberRemove(event *gateway.GuildMemberRemoveEvent) {
  167. v.messageInput.cache.Invalidate(event.GuildID.String()+" "+event.User.Username, v.state.MemberState.SearchLimit)
  168. }
  169. func (v *View) onTypingStart(event *gateway.TypingStartEvent) {
  170. selectedChannel := v.SelectedChannel()
  171. if selectedChannel == nil {
  172. return
  173. }
  174. if selectedChannel.ID != event.ChannelID {
  175. return
  176. }
  177. me, _ := v.state.Cabinet.Me()
  178. if event.UserID == me.ID {
  179. return
  180. }
  181. v.addTyper(event.UserID)
  182. }