state.go 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. package chat
  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. )
  18. func (v *View) OpenState(token string) error {
  19. identifyProps := http.IdentifyProperties()
  20. gateway.DefaultIdentity = identifyProps
  21. gateway.DefaultPresence = &gateway.UpdatePresenceCommand{
  22. Status: v.cfg.Status,
  23. }
  24. id := gateway.DefaultIdentifier(token)
  25. id.Compress = false
  26. session := session.NewCustom(id, http.NewClient(token), handler.New())
  27. state := state.NewFromSession(session, defaultstore.New())
  28. v.state = ningen.FromState(state)
  29. // Handlers
  30. v.state.AddHandler(v.onRaw)
  31. v.state.AddHandler(v.onReady)
  32. v.state.AddHandler(v.onMessageCreate)
  33. v.state.AddHandler(v.onMessageUpdate)
  34. v.state.AddHandler(v.onMessageDelete)
  35. v.state.AddHandler(v.onReadUpdate)
  36. v.state.AddHandler(v.onGuildMembersChunk)
  37. v.state.AddHandler(v.onGuildMemberRemove)
  38. if v.cfg.TypingIndicator.Receive {
  39. v.state.AddHandler(v.onTypingStart)
  40. }
  41. v.state.StateLog = func(err error) {
  42. slog.Error("state log", "err", err)
  43. }
  44. v.state.OnRequest = append(v.state.OnRequest, httputil.WithHeaders(http.Headers()), v.onRequest)
  45. return v.state.Open(context.TODO())
  46. }
  47. func (v *View) CloseState() error {
  48. if v.state == nil {
  49. return nil
  50. }
  51. return v.state.Close()
  52. }
  53. func (v *View) onRequest(r httpdriver.Request) error {
  54. if req, ok := r.(*httpdriver.DefaultRequest); ok {
  55. slog.Debug("new HTTP request", "method", req.Method, "url", req.URL)
  56. }
  57. return nil
  58. }
  59. func (v *View) onRaw(event *ws.RawEvent) {
  60. slog.Debug(
  61. "new raw event",
  62. "code", event.OriginalCode,
  63. "type", event.OriginalType,
  64. // "data", event.Raw,
  65. )
  66. }
  67. func (v *View) onReady(r *gateway.ReadyEvent) {
  68. dmNode := tview.NewTreeNode("Direct Messages")
  69. root := v.guildsTree.
  70. GetRoot().
  71. ClearChildren().
  72. AddChild(dmNode)
  73. for _, folder := range r.UserSettings.GuildFolders {
  74. if folder.ID == 0 && len(folder.GuildIDs) == 1 {
  75. guild, err := v.state.Cabinet.Guild(folder.GuildIDs[0])
  76. if err != nil {
  77. slog.Error(
  78. "failed to get guild from state",
  79. "guild_id",
  80. folder.GuildIDs[0],
  81. "err",
  82. err,
  83. )
  84. continue
  85. }
  86. v.guildsTree.createGuildNode(root, *guild)
  87. } else {
  88. v.guildsTree.createFolderNode(folder)
  89. }
  90. }
  91. v.guildsTree.SetCurrentNode(root)
  92. v.app.SetFocus(v.guildsTree)
  93. v.app.Draw()
  94. }
  95. func (v *View) onMessageCreate(message *gateway.MessageCreateEvent) {
  96. selectedChannel := v.SelectedChannel()
  97. if selectedChannel != nil && selectedChannel.ID == message.ChannelID {
  98. v.removeTyper(message.Author.ID)
  99. v.messagesList.drawMessage(v.messagesList, message.Message)
  100. v.app.Draw()
  101. } else {
  102. if err := notifications.Notify(v.state, message, v.cfg); err != nil {
  103. slog.Error("failed to notify", "err", err, "channel_id", message.ChannelID, "message_id", message.ID)
  104. }
  105. }
  106. }
  107. func (v *View) onMessageUpdate(message *gateway.MessageUpdateEvent) {
  108. if selected := v.SelectedChannel(); selected != nil && selected.ID == message.ChannelID {
  109. v.onMessageDelete(&gateway.MessageDeleteEvent{ID: message.ID, ChannelID: message.ChannelID, GuildID: message.GuildID})
  110. }
  111. }
  112. func (v *View) onMessageDelete(message *gateway.MessageDeleteEvent) {
  113. if selected := v.SelectedChannel(); selected != nil && selected.ID == message.ChannelID {
  114. messages, err := v.state.Cabinet.Messages(message.ChannelID)
  115. if err != nil {
  116. slog.Error("failed to get messages from state", "err", err, "channel_id", message.ChannelID)
  117. return
  118. }
  119. v.messagesList.reset()
  120. v.messagesList.drawMessages(messages)
  121. v.app.Draw()
  122. }
  123. }
  124. func (v *View) onGuildMembersChunk(event *gateway.GuildMembersChunkEvent) {
  125. v.messagesList.setFetchingChunk(false, uint(len(event.Members)))
  126. }
  127. func (v *View) onGuildMemberRemove(event *gateway.GuildMemberRemoveEvent) {
  128. v.messageInput.cache.Invalidate(event.GuildID.String()+" "+event.User.Username, v.state.MemberState.SearchLimit)
  129. }
  130. func (v *View) onTypingStart(event *gateway.TypingStartEvent) {
  131. selectedChannel := v.SelectedChannel()
  132. if selectedChannel == nil {
  133. return
  134. }
  135. if selectedChannel.ID != event.ChannelID {
  136. return
  137. }
  138. me, err := v.state.Cabinet.Me()
  139. if err != nil {
  140. slog.Error("failed to get me from state", "err", err)
  141. return
  142. }
  143. if event.UserID == me.ID {
  144. return
  145. }
  146. v.addTyper(event.UserID)
  147. }