guilds_tree.go 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. package cmd
  2. import (
  3. "fmt"
  4. "log"
  5. "slices"
  6. "sort"
  7. "strings"
  8. "github.com/ayn2op/discordo/pkg/itertools"
  9. "github.com/diamondburned/arikawa/v3/discord"
  10. "github.com/diamondburned/arikawa/v3/gateway"
  11. "github.com/gdamore/tcell/v2"
  12. "github.com/rivo/tview"
  13. )
  14. type GuildsTree struct {
  15. *tview.TreeView
  16. selectedChannelID discord.ChannelID
  17. }
  18. func newGuildsTree() *GuildsTree {
  19. gt := &GuildsTree{
  20. TreeView: tview.NewTreeView(),
  21. }
  22. root := tview.NewTreeNode("")
  23. gt.SetRoot(root)
  24. gt.SetTopLevel(1)
  25. gt.SetGraphics(cfg.Theme.GuildsTree.Graphics)
  26. gt.SetBackgroundColor(tcell.GetColor(cfg.Theme.BackgroundColor))
  27. gt.SetSelectedFunc(gt.onSelected)
  28. gt.SetTitle("Guilds")
  29. gt.SetTitleColor(tcell.GetColor(cfg.Theme.TitleColor))
  30. gt.SetTitleAlign(tview.AlignLeft)
  31. p := cfg.Theme.BorderPadding
  32. gt.SetBorder(cfg.Theme.Border)
  33. gt.SetBorderColor(tcell.GetColor(cfg.Theme.BorderColor))
  34. gt.SetBorderPadding(p[0], p[1], p[2], p[3])
  35. gt.SetInputCapture(gt.onInputCapture)
  36. return gt
  37. }
  38. func (gt *GuildsTree) createFolderNode(folder gateway.GuildFolder) {
  39. var name string
  40. if folder.Name == "" {
  41. name = "Folder"
  42. } else {
  43. name = fmt.Sprintf("[%s]%s[-]", folder.Color.String(), folder.Name)
  44. }
  45. root := gt.GetRoot()
  46. folderNode := tview.NewTreeNode(name)
  47. folderNode.SetExpanded(cfg.Theme.GuildsTree.AutoExpandFolders)
  48. root.AddChild(folderNode)
  49. for _, gID := range folder.GuildIDs {
  50. g, err := discordState.Cabinet.Guild(gID)
  51. if err != nil {
  52. log.Printf("guild %v not found in state: %v\n", gID, err)
  53. continue
  54. }
  55. gt.createGuildNode(folderNode, *g)
  56. }
  57. }
  58. func (gt *GuildsTree) createGuildNode(n *tview.TreeNode, g discord.Guild) {
  59. guildNode := tview.NewTreeNode(g.Name)
  60. guildNode.SetReference(g.ID)
  61. n.AddChild(guildNode)
  62. }
  63. func (gt *GuildsTree) channelToString(c discord.Channel) string {
  64. var s string
  65. switch c.Type {
  66. case discord.GuildText:
  67. s = "#" + c.Name
  68. case discord.DirectMessage:
  69. r := c.DMRecipients[0]
  70. s = r.Tag()
  71. case discord.GuildVoice:
  72. s = "v-" + c.Name
  73. case discord.GroupDM:
  74. s = c.Name
  75. // If the name of the channel is empty, use the recipients' tags
  76. if s == "" {
  77. recipients := itertools.Map(slices.Values(c.DMRecipients), func(r discord.User) string {
  78. return r.Tag()
  79. })
  80. s = strings.Join(slices.Collect(recipients), ", ")
  81. }
  82. case discord.GuildAnnouncement:
  83. s = "a-" + c.Name
  84. case discord.GuildStore:
  85. s = "s-" + c.Name
  86. case discord.GuildForum:
  87. s = "f-" + c.Name
  88. default:
  89. s = c.Name
  90. }
  91. return s
  92. }
  93. func (gt *GuildsTree) createChannelNode(n *tview.TreeNode, c discord.Channel) *tview.TreeNode {
  94. if c.Type != discord.DirectMessage && c.Type != discord.GroupDM {
  95. ps, err := discordState.Permissions(c.ID, discordState.Ready().User.ID)
  96. if err != nil {
  97. log.Println(err)
  98. return nil
  99. }
  100. if !ps.Has(discord.PermissionViewChannel) {
  101. return nil
  102. }
  103. }
  104. channelNode := tview.NewTreeNode(gt.channelToString(c))
  105. channelNode.SetReference(c.ID)
  106. n.AddChild(channelNode)
  107. return channelNode
  108. }
  109. func (gt *GuildsTree) createChannelNodes(n *tview.TreeNode, cs []discord.Channel) {
  110. orphanChannels := itertools.Filter(slices.Values(cs), func(c discord.Channel) bool {
  111. return c.Type != discord.GuildCategory && !c.ParentID.IsValid()
  112. })
  113. for _, c := range slices.Collect(orphanChannels) {
  114. gt.createChannelNode(n, c)
  115. }
  116. PARENT_CHANNELS:
  117. for _, c := range cs {
  118. if c.Type == discord.GuildCategory {
  119. for _, nested := range cs {
  120. if nested.ParentID == c.ID {
  121. gt.createChannelNode(n, c)
  122. continue PARENT_CHANNELS
  123. }
  124. }
  125. }
  126. }
  127. for _, c := range cs {
  128. if c.ParentID.IsValid() {
  129. var parent *tview.TreeNode
  130. n.Walk(func(node, _ *tview.TreeNode) bool {
  131. if node.GetReference() == c.ParentID {
  132. parent = node
  133. return false
  134. }
  135. return true
  136. })
  137. if parent != nil {
  138. gt.createChannelNode(parent, c)
  139. }
  140. }
  141. }
  142. }
  143. func (gt *GuildsTree) onSelected(n *tview.TreeNode) {
  144. gt.selectedChannelID = 0
  145. mainFlex.messagesText.reset()
  146. mainFlex.messageInput.reset()
  147. if len(n.GetChildren()) != 0 {
  148. n.SetExpanded(!n.IsExpanded())
  149. return
  150. }
  151. switch ref := n.GetReference().(type) {
  152. case discord.GuildID:
  153. cs, err := discordState.Cabinet.Channels(ref)
  154. if err != nil {
  155. log.Println(err)
  156. return
  157. }
  158. sort.Slice(cs, func(i, j int) bool {
  159. return cs[i].Position < cs[j].Position
  160. })
  161. gt.createChannelNodes(n, cs)
  162. case discord.ChannelID:
  163. mainFlex.messagesText.drawMsgs(ref)
  164. mainFlex.messagesText.ScrollToEnd()
  165. c, err := discordState.Cabinet.Channel(ref)
  166. if err != nil {
  167. log.Println(err)
  168. return
  169. }
  170. mainFlex.messagesText.SetTitle(gt.channelToString(*c))
  171. gt.selectedChannelID = ref
  172. app.SetFocus(mainFlex.messageInput)
  173. case nil: // Direct messages
  174. cs, err := discordState.Cabinet.PrivateChannels()
  175. if err != nil {
  176. log.Println(err)
  177. return
  178. }
  179. sort.Slice(cs, func(i, j int) bool {
  180. return cs[i].LastMessageID > cs[j].LastMessageID
  181. })
  182. for _, c := range cs {
  183. gt.createChannelNode(n, c)
  184. }
  185. }
  186. }
  187. func (gt *GuildsTree) onInputCapture(event *tcell.EventKey) *tcell.EventKey {
  188. switch event.Name() {
  189. case cfg.Keys.SelectPrevious:
  190. return tcell.NewEventKey(tcell.KeyUp, 0, tcell.ModNone)
  191. case cfg.Keys.SelectNext:
  192. return tcell.NewEventKey(tcell.KeyDown, 0, tcell.ModNone)
  193. case cfg.Keys.SelectFirst:
  194. return tcell.NewEventKey(tcell.KeyHome, 0, tcell.ModNone)
  195. case cfg.Keys.SelectLast:
  196. return tcell.NewEventKey(tcell.KeyEnd, 0, tcell.ModNone)
  197. case cfg.Keys.GuildsTree.SelectCurrent:
  198. return tcell.NewEventKey(tcell.KeyEnter, 0, tcell.ModNone)
  199. }
  200. return nil
  201. }