state.go 6.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235
  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 (m *Model) OpenState(token string) error {
  21. identifyProps := http.IdentifyProperties()
  22. gateway.DefaultIdentity = identifyProps
  23. gateway.DefaultPresence = &gateway.UpdatePresenceCommand{
  24. Status: m.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. m.state = ningen.FromState(state)
  31. // Handlers
  32. m.state.AddHandler(m.onRaw)
  33. m.state.AddHandler(m.onReady)
  34. m.state.AddHandler(m.onMessageCreate)
  35. m.state.AddHandler(m.onMessageUpdate)
  36. m.state.AddHandler(m.onMessageDelete)
  37. m.state.AddHandler(m.onReadUpdate)
  38. m.state.AddHandler(m.onGuildMembersChunk)
  39. m.state.AddHandler(m.onGuildMemberRemove)
  40. if m.cfg.TypingIndicator.Receive {
  41. m.state.AddHandler(m.onTypingStart)
  42. }
  43. m.state.StateLog = func(err error) {
  44. slog.Error("state log", "err", err)
  45. }
  46. m.state.OnRequest = append(m.state.OnRequest, httputil.WithHeaders(http.Headers()), m.onRequest)
  47. return m.state.Open(context.Background())
  48. }
  49. func (m *Model) 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 (m *Model) 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 (m *Model) onReady(event *gateway.ReadyEvent) {
  64. m.app.QueueUpdateDraw(func() {
  65. // Rebuild indexes from scratch so reconnects and account switches do not
  66. // retain pointers to detached tree nodes.
  67. m.guildsTree.resetNodeIndex()
  68. dmNode := tview.NewTreeNode("Direct Messages").SetReference(dmNode{}).SetExpandable(true).SetExpanded(false)
  69. m.guildsTree.dmRootNode = dmNode
  70. root := m.guildsTree.
  71. GetRoot().
  72. ClearChildren().
  73. AddChild(dmNode)
  74. // Track guilds already in folders to find orphans (newly joined guilds may not be synced to GuildFolders yet but always appear in GuildPositions)
  75. guildsInFolders := make(map[discord.GuildID]bool)
  76. for _, folder := range event.UserSettings.GuildFolders {
  77. for _, guildID := range folder.GuildIDs {
  78. guildsInFolders[guildID] = true
  79. }
  80. }
  81. // Build index of all available guilds.
  82. guildsByID := make(map[discord.GuildID]*gateway.GuildCreateEvent, len(event.Guilds))
  83. for index := range event.Guilds {
  84. guildsByID[event.Guilds[index].ID] = &event.Guilds[index]
  85. }
  86. // Use GuildPositions for ordering (it's the canonical order).
  87. // Guilds not in any folder are "orphans" - add them directly to root.
  88. positions := event.UserSettings.GuildPositions
  89. // Fallback: GuildPositions shouldn't be nil but handle gracefully
  90. if len(positions) == 0 {
  91. positions = make([]discord.GuildID, 0, len(event.Guilds))
  92. for _, guildEvent := range event.Guilds {
  93. positions = append(positions, guildEvent.ID)
  94. }
  95. }
  96. for _, guildID := range positions {
  97. // Already handled in folder processing below
  98. if guildsInFolders[guildID] {
  99. continue
  100. }
  101. // Orphan guild - add directly to root in order
  102. if guildEvent, ok := guildsByID[guildID]; ok {
  103. m.guildsTree.createGuildNode(root, guildEvent.Guild)
  104. }
  105. }
  106. // Process folders (real folders and single-guild "folders")
  107. for _, folder := range event.UserSettings.GuildFolders {
  108. if folder.ID == 0 && len(folder.GuildIDs) == 1 {
  109. if guild, ok := guildsByID[folder.GuildIDs[0]]; ok {
  110. m.guildsTree.createGuildNode(root, guild.Guild)
  111. }
  112. } else {
  113. m.guildsTree.createFolderNode(folder, guildsByID)
  114. }
  115. }
  116. m.guildsTree.SetCurrentNode(root)
  117. m.app.SetFocus(m.guildsTree)
  118. })
  119. }
  120. func (m *Model) onMessageCreate(message *gateway.MessageCreateEvent) {
  121. selectedChannel := m.SelectedChannel()
  122. if selectedChannel != nil && selectedChannel.ID == message.ChannelID {
  123. m.removeTyper(message.Author.ID)
  124. m.app.QueueUpdateDraw(func() {
  125. m.messagesList.addMessage(message.Message)
  126. })
  127. } else {
  128. if err := notifications.Notify(m.state, message, m.cfg); err != nil {
  129. slog.Error("failed to notify", "err", err, "channel_id", message.ChannelID, "message_id", message.ID)
  130. }
  131. }
  132. }
  133. func (m *Model) onMessageUpdate(message *gateway.MessageUpdateEvent) {
  134. if selected := m.SelectedChannel(); selected != nil && selected.ID == message.ChannelID {
  135. index := slices.IndexFunc(m.messagesList.messages, func(m discord.Message) bool {
  136. return m.ID == message.ID
  137. })
  138. if index < 0 {
  139. return
  140. }
  141. m.app.QueueUpdateDraw(func() {
  142. m.messagesList.setMessage(index, message.Message)
  143. })
  144. }
  145. }
  146. func (m *Model) onMessageDelete(message *gateway.MessageDeleteEvent) {
  147. if selected := m.SelectedChannel(); selected != nil && selected.ID == message.ChannelID {
  148. prevCursor := m.messagesList.Cursor()
  149. deletedIndex := slices.IndexFunc(m.messagesList.messages, func(m discord.Message) bool {
  150. return m.ID == message.ID
  151. })
  152. if deletedIndex < 0 {
  153. return
  154. }
  155. m.app.QueueUpdateDraw(func() {
  156. m.messagesList.deleteMessage(deletedIndex)
  157. })
  158. // Keep cursor stable when possible after removal.
  159. newCursor := prevCursor
  160. if prevCursor == deletedIndex {
  161. // Prefer previous item; fall forward if we deleted the first.
  162. newCursor = deletedIndex - 1
  163. if newCursor < 0 {
  164. if deletedIndex < len(m.messagesList.messages) {
  165. newCursor = deletedIndex
  166. } else {
  167. newCursor = -1
  168. }
  169. }
  170. } else if prevCursor > deletedIndex {
  171. // Shift back since the list shrank before the cursor.
  172. newCursor = prevCursor - 1
  173. }
  174. if newCursor != prevCursor {
  175. // Avoid redundant cursor updates if nothing changed.
  176. m.app.QueueUpdateDraw(func() {
  177. m.messagesList.SetCursor(newCursor)
  178. })
  179. }
  180. }
  181. }
  182. func (m *Model) onGuildMembersChunk(event *gateway.GuildMembersChunkEvent) {
  183. m.messagesList.setFetchingChunk(false, uint(len(event.Members)))
  184. }
  185. func (m *Model) onGuildMemberRemove(event *gateway.GuildMemberRemoveEvent) {
  186. m.messageInput.cache.Invalidate(event.GuildID.String()+" "+event.User.Username, m.state.MemberState.SearchLimit)
  187. }
  188. func (m *Model) onTypingStart(event *gateway.TypingStartEvent) {
  189. selectedChannel := m.SelectedChannel()
  190. if selectedChannel == nil {
  191. return
  192. }
  193. if selectedChannel.ID != event.ChannelID {
  194. return
  195. }
  196. me, _ := m.state.Cabinet.Me()
  197. if event.UserID == me.ID {
  198. return
  199. }
  200. m.addTyper(event.UserID)
  201. }