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