guilds_tree.go 3.8 KB

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