guilds_tree.go 3.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. package main
  2. import (
  3. "log"
  4. "sort"
  5. "strings"
  6. "github.com/diamondburned/arikawa/v3/discord"
  7. "github.com/gdamore/tcell/v2"
  8. "github.com/rivo/tview"
  9. )
  10. type GuildsTree struct {
  11. *tview.TreeView
  12. root *tview.TreeNode
  13. selectedChannel *discord.Channel
  14. }
  15. func newGuildsTree() *GuildsTree {
  16. gt := &GuildsTree{
  17. TreeView: tview.NewTreeView(),
  18. root: tview.NewTreeNode(""),
  19. }
  20. gt.SetGraphics(cfg.Theme.GuildsTree.Graphics)
  21. gt.SetRoot(gt.root)
  22. gt.SetTopLevel(1)
  23. gt.SetSelectedFunc(gt.onSelected)
  24. gt.SetTitle("Guilds")
  25. gt.SetTitleColor(tcell.GetColor(cfg.Theme.MessagesText.TitleColor))
  26. gt.SetTitleAlign(tview.AlignLeft)
  27. padding := cfg.Theme.GuildsTree.BorderPadding
  28. gt.SetBorder(cfg.Theme.GuildsTree.Border)
  29. gt.SetBorderPadding(padding[0], padding[1], padding[2], padding[3])
  30. return gt
  31. }
  32. func (gt *GuildsTree) newGuildFromID(n *tview.TreeNode, gid discord.GuildID) error {
  33. g, err := discordState.Cabinet.Guild(gid)
  34. if err != nil {
  35. return err
  36. }
  37. gn := tview.NewTreeNode(g.Name)
  38. gn.SetReference(g.ID)
  39. n.AddChild(gn)
  40. return nil
  41. }
  42. func (gt *GuildsTree) newChannel(n *tview.TreeNode, c discord.Channel) {
  43. cn := tview.NewTreeNode(gt.channelToString(c))
  44. cn.SetReference(c.ID)
  45. n.AddChild(cn)
  46. }
  47. func (gt *GuildsTree) channelToString(c discord.Channel) string {
  48. var s string
  49. switch c.Type {
  50. case discord.GuildText:
  51. s = "#" + c.Name
  52. case discord.DirectMessage:
  53. r := c.DMRecipients[0]
  54. s = r.Tag()
  55. case discord.GuildVoice:
  56. s = "v-" + c.Name
  57. case discord.GroupDM:
  58. s = c.Name
  59. // If the name of the channel is empty, use the recipients tags
  60. if s == "" {
  61. rs := make([]string, len(c.DMRecipients))
  62. for _, r := range c.DMRecipients {
  63. rs = append(rs, r.Tag())
  64. }
  65. s = strings.Join(rs, ", ")
  66. }
  67. case discord.GuildNews:
  68. s = "n-" + c.Name
  69. case discord.GuildStore:
  70. s = "s-" + c.Name
  71. default:
  72. s = c.Name
  73. }
  74. return s
  75. }
  76. func (gt *GuildsTree) onSelected(n *tview.TreeNode) {
  77. gt.selectedChannel = nil
  78. messagesText.selectedMessage = nil
  79. messagesText.Highlight()
  80. messageInput.SetTitle("")
  81. messageInput.SetText("")
  82. if len(n.GetChildren()) != 0 {
  83. n.SetExpanded(!n.IsExpanded())
  84. return
  85. }
  86. switch ref := n.GetReference().(type) {
  87. case discord.GuildID:
  88. cs, err := discordState.Cabinet.Channels(ref)
  89. if err != nil {
  90. log.Println(err)
  91. return
  92. }
  93. sort.Slice(cs, func(i, j int) bool {
  94. return cs[i].Position < cs[j].Position
  95. })
  96. // Orphan (top-level) channels
  97. for _, c := range cs {
  98. if c.Type != discord.GuildCategory && !c.ParentID.IsValid() {
  99. gt.newChannel(n, c)
  100. }
  101. }
  102. // Category channels
  103. CATEGORY:
  104. for _, c := range cs {
  105. if c.Type == discord.GuildCategory {
  106. for _, nestedChannel := range cs {
  107. if nestedChannel.ParentID == c.ID {
  108. gt.newChannel(n, c)
  109. continue CATEGORY
  110. }
  111. }
  112. gt.newChannel(n, c)
  113. }
  114. }
  115. // Children (category-bound) channels
  116. for _, c := range cs {
  117. if c.Type != discord.GuildCategory && c.ParentID.IsValid() {
  118. var parent *tview.TreeNode
  119. n.Walk(func(node, _ *tview.TreeNode) bool {
  120. if node.GetReference() == c.ParentID {
  121. parent = node
  122. return false
  123. }
  124. return true
  125. })
  126. if parent != nil {
  127. gt.newChannel(parent, c)
  128. }
  129. }
  130. }
  131. case discord.ChannelID:
  132. messagesText.Clear()
  133. c, err := discordState.Cabinet.Channel(ref)
  134. if err != nil {
  135. log.Println(err)
  136. return
  137. }
  138. gt.selectedChannel = c
  139. messagesText.SetTitle(gt.channelToString(*c))
  140. ms, err := discordState.Messages(ref, cfg.MessagesLimit)
  141. if err != nil {
  142. log.Println(err)
  143. return
  144. }
  145. for i := len(ms) - 1; i >= 0; i-- {
  146. if messagesText.newMessage(&ms[i]); err != nil {
  147. log.Println(err)
  148. continue
  149. }
  150. }
  151. app.SetFocus(messagesText)
  152. case nil: // Direct messages
  153. cs, err := discordState.Cabinet.PrivateChannels()
  154. if err != nil {
  155. log.Println(err)
  156. return
  157. }
  158. sort.Slice(cs, func(i, j int) bool {
  159. return cs[i].LastMessageID < cs[j].LastMessageID
  160. })
  161. for _, c := range cs {
  162. gt.newChannel(n, c)
  163. }
  164. }
  165. }