ui.go 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399
  1. package main
  2. import (
  3. "sort"
  4. "strings"
  5. "github.com/atotto/clipboard"
  6. "github.com/ayntgl/discordgo"
  7. "github.com/ayntgl/discordo/util"
  8. "github.com/gdamore/tcell/v2"
  9. "github.com/rivo/tview"
  10. )
  11. var (
  12. selectedChannel *discordgo.Channel
  13. selectedMessage int = -1
  14. )
  15. func onAppInputCapture(e *tcell.EventKey) *tcell.EventKey {
  16. if hasKeybinding(conf.Keybindings.FocusChannelsTree, e.Name()) {
  17. app.SetFocus(channelsTree)
  18. return nil
  19. } else if hasKeybinding(conf.Keybindings.FocusMessagesView, e.Name()) {
  20. app.SetFocus(messagesView)
  21. return nil
  22. } else if hasKeybinding(conf.Keybindings.FocusMessageInputField, e.Name()) {
  23. app.SetFocus(messageInputField)
  24. return nil
  25. }
  26. return e
  27. }
  28. func onChannelsTreeSelected(n *tview.TreeNode) {
  29. selectedChannel = nil
  30. selectedMessage = 0
  31. messagesView.
  32. Clear().
  33. SetTitle("")
  34. messageInputField.SetText("")
  35. // Unhighlight the already-highlighted regions.
  36. messagesView.Highlight()
  37. id := n.GetReference()
  38. switch n.GetLevel() {
  39. case 1: // Guilds or Direct Messages
  40. if len(n.GetChildren()) == 0 {
  41. // If the reference of the selected `*TreeNode` is `nil`, it is the direct messages `*TreeNode`.
  42. if id == nil {
  43. cs := session.State.PrivateChannels
  44. sort.Slice(cs, func(i, j int) bool {
  45. return cs[i].LastMessageID > cs[j].LastMessageID
  46. })
  47. for _, c := range cs {
  48. tag := "[::d]"
  49. if channelIsUnread(session.State, c) {
  50. tag = "[::b]"
  51. }
  52. cn := tview.NewTreeNode(tag + util.ChannelToString(c) + "[::-]").
  53. SetReference(c.ID).
  54. Collapse()
  55. n.AddChild(cn)
  56. }
  57. } else {
  58. g, err := session.State.Guild(id.(string))
  59. if err != nil {
  60. return
  61. }
  62. sort.Slice(g.Channels, func(i, j int) bool {
  63. return g.Channels[i].Position < g.Channels[j].Position
  64. })
  65. // Top-level channels
  66. createTopLevelChannelsNodes(channelsTree, session.State, n, g.Channels)
  67. // Category channels
  68. createCategoryChannelsNodes(channelsTree, session.State, n, g.Channels)
  69. // Second-level channels
  70. createSecondLevelChannelsNodes(channelsTree, session.State, g.Channels)
  71. }
  72. }
  73. n.SetExpanded(!n.IsExpanded())
  74. default: // Channels
  75. c, err := session.State.Channel(id.(string))
  76. if err != nil {
  77. return
  78. }
  79. selectedChannel = c
  80. app.SetFocus(messageInputField)
  81. switch c.Type {
  82. case discordgo.ChannelTypeGuildText, discordgo.ChannelTypeGuildNews:
  83. title := util.ChannelToString(c)
  84. if c.Topic != "" {
  85. title += " - " + c.Topic
  86. }
  87. messagesView.SetTitle(title)
  88. case discordgo.ChannelTypeDM, discordgo.ChannelTypeGroupDM:
  89. messagesView.SetTitle(util.ChannelToString(c))
  90. }
  91. if strings.HasPrefix(n.GetText(), "[::b]") {
  92. n.SetText("[::d]" + util.ChannelToString(c) + "[::-]")
  93. }
  94. go func() {
  95. ms, err := session.ChannelMessages(c.ID, conf.GetMessagesLimit, "", "", "")
  96. if err != nil {
  97. return
  98. }
  99. for i := len(ms) - 1; i >= 0; i-- {
  100. selectedChannel.Messages = append(selectedChannel.Messages, ms[i])
  101. messagesView.Write(buildMessage(ms[i]))
  102. }
  103. // Scroll to the end of the text after the messages have been written to the TextView.
  104. messagesView.ScrollToEnd()
  105. if len(ms) != 0 && channelIsUnread(session.State, c) {
  106. session.ChannelMessageAck(c.ID, c.LastMessageID, "")
  107. }
  108. }()
  109. }
  110. }
  111. // createTopLevelChannelsNodes builds and creates `*tview.TreeNode`s for top-level (channels that have an empty parent ID and of type GUILD_TEXT, GUILD_NEWS) channels. If the client user does not have the VIEW_CHANNEL permission for a channel, the channel is excluded from the parent.
  112. func createTopLevelChannelsNodes(treeView *tview.TreeView, s *discordgo.State, n *tview.TreeNode, cs []*discordgo.Channel) {
  113. for _, c := range cs {
  114. if (c.Type == discordgo.ChannelTypeGuildText || c.Type == discordgo.ChannelTypeGuildNews) &&
  115. (c.ParentID == "") {
  116. if !hasPermission(s, c.ID, discordgo.PermissionViewChannel) {
  117. continue
  118. }
  119. n.AddChild(createChannelNode(s, c))
  120. continue
  121. }
  122. }
  123. }
  124. // createCategoryChannelsNodes builds and creates `*tview.TreeNode`s for category (type: GUILD_CATEGORY) channels. If the client user does not have the VIEW_CHANNEL permission for a channel, the channel is excluded from the parent.
  125. func createCategoryChannelsNodes(treeView *tview.TreeView, s *discordgo.State, n *tview.TreeNode, cs []*discordgo.Channel) {
  126. CategoryLoop:
  127. for _, c := range cs {
  128. if c.Type == discordgo.ChannelTypeGuildCategory {
  129. if !hasPermission(s, c.ID, discordgo.PermissionViewChannel) {
  130. continue
  131. }
  132. for _, child := range cs {
  133. if child.ParentID == c.ID {
  134. n.AddChild(createChannelNode(s, c))
  135. continue CategoryLoop
  136. }
  137. }
  138. n.AddChild(createChannelNode(s, c))
  139. }
  140. }
  141. }
  142. // createSecondLevelChannelsNodes builds and creates `*tview.TreeNode`s for second-level (channels that have a non-empty parent ID and of type GUILD_TEXT, GUILD_NEWS) channels. If the client user does not have the VIEW_CHANNEL permission for a channel, the channel is excluded from the parent.
  143. func createSecondLevelChannelsNodes(treeView *tview.TreeView, s *discordgo.State, cs []*discordgo.Channel) {
  144. for _, c := range cs {
  145. if (c.Type == discordgo.ChannelTypeGuildText || c.Type == discordgo.ChannelTypeGuildNews) &&
  146. (c.ParentID != "") {
  147. if !hasPermission(s, c.ID, discordgo.PermissionViewChannel) {
  148. continue
  149. }
  150. pn := util.GetNodeByReference(treeView, c.ParentID)
  151. if pn != nil {
  152. pn.AddChild(createChannelNode(s, c))
  153. }
  154. }
  155. }
  156. }
  157. func onMessagesViewInputCapture(e *tcell.EventKey) *tcell.EventKey {
  158. if selectedChannel == nil {
  159. return nil
  160. }
  161. ms := selectedChannel.Messages
  162. if len(ms) == 0 {
  163. return nil
  164. }
  165. if hasKeybinding(conf.Keybindings.SelectPreviousMessage, e.Name()) {
  166. if len(messagesView.GetHighlights()) == 0 {
  167. selectedMessage = len(ms) - 1
  168. } else {
  169. selectedMessage--
  170. if selectedMessage < 0 {
  171. selectedMessage = 0
  172. }
  173. }
  174. messagesView.
  175. Highlight(ms[selectedMessage].ID).
  176. ScrollToHighlight()
  177. return nil
  178. } else if hasKeybinding(conf.Keybindings.SelectNextMessage, e.Name()) {
  179. if len(messagesView.GetHighlights()) == 0 {
  180. selectedMessage = len(ms) - 1
  181. } else {
  182. selectedMessage++
  183. if selectedMessage >= len(ms) {
  184. selectedMessage = len(ms) - 1
  185. }
  186. }
  187. messagesView.
  188. Highlight(ms[selectedMessage].ID).
  189. ScrollToHighlight()
  190. return nil
  191. } else if hasKeybinding(conf.Keybindings.SelectFirstMessage, e.Name()) {
  192. selectedMessage = 0
  193. messagesView.
  194. Highlight(ms[selectedMessage].ID).
  195. ScrollToHighlight()
  196. return nil
  197. } else if hasKeybinding(conf.Keybindings.SelectLastMessage, e.Name()) {
  198. selectedMessage = len(ms) - 1
  199. messagesView.
  200. Highlight(ms[selectedMessage].ID).
  201. ScrollToHighlight()
  202. return nil
  203. } else if hasKeybinding(conf.Keybindings.SelectMessageReference, e.Name()) {
  204. hs := messagesView.GetHighlights()
  205. if len(hs) == 0 {
  206. return nil
  207. }
  208. _, m := findMessageByID(selectedChannel.Messages, hs[0])
  209. if m.ReferencedMessage != nil {
  210. selectedMessage, _ = findMessageByID(selectedChannel.Messages, m.ReferencedMessage.ID)
  211. messagesView.
  212. Highlight(m.ReferencedMessage.ID).
  213. ScrollToHighlight()
  214. }
  215. return nil
  216. } else if hasKeybinding(conf.Keybindings.ReplySelectedMessage, e.Name()) {
  217. hs := messagesView.GetHighlights()
  218. if len(hs) == 0 {
  219. return nil
  220. }
  221. _, m := findMessageByID(selectedChannel.Messages, hs[0])
  222. messageInputField.SetTitle("Replying to " + m.Author.String())
  223. app.SetFocus(messageInputField)
  224. return nil
  225. } else if hasKeybinding(conf.Keybindings.MentionReplySelectedMessage, e.Name()) {
  226. hs := messagesView.GetHighlights()
  227. if len(hs) == 0 {
  228. return nil
  229. }
  230. _, m := findMessageByID(selectedChannel.Messages, hs[0])
  231. messageInputField.SetTitle("[@] Replying to " + m.Author.String())
  232. app.SetFocus(messageInputField)
  233. return nil
  234. } else if hasKeybinding(conf.Keybindings.CopySelectedMessage, e.Name()) {
  235. hs := messagesView.GetHighlights()
  236. if len(hs) == 0 {
  237. return nil
  238. }
  239. _, m := findMessageByID(selectedChannel.Messages, hs[0])
  240. err := clipboard.WriteAll(m.Content)
  241. if err != nil {
  242. return nil
  243. }
  244. return nil
  245. }
  246. return e
  247. }
  248. func onMessageInputFieldInputCapture(e *tcell.EventKey) *tcell.EventKey {
  249. switch e.Key() {
  250. case tcell.KeyEnter:
  251. if selectedChannel == nil {
  252. return nil
  253. }
  254. t := strings.TrimSpace(messageInputField.GetText())
  255. if t == "" {
  256. return nil
  257. }
  258. if len(messagesView.GetHighlights()) != 0 {
  259. m := selectedChannel.Messages[selectedMessage]
  260. d := &discordgo.MessageSend{
  261. Content: t,
  262. Reference: m.Reference(),
  263. AllowedMentions: &discordgo.MessageAllowedMentions{RepliedUser: false},
  264. }
  265. if strings.HasPrefix(messageInputField.GetTitle(), "[@]") {
  266. d.AllowedMentions.RepliedUser = true
  267. } else {
  268. d.AllowedMentions.RepliedUser = false
  269. }
  270. go session.ChannelMessageSendComplex(m.ChannelID, d)
  271. selectedMessage = -1
  272. messagesView.Highlight()
  273. messageInputField.SetTitle("")
  274. } else {
  275. go session.ChannelMessageSend(selectedChannel.ID, t)
  276. }
  277. messageInputField.SetText("")
  278. return nil
  279. case tcell.KeyCtrlV:
  280. text, _ := clipboard.ReadAll()
  281. text = messageInputField.GetText() + text
  282. messageInputField.SetText(text)
  283. return nil
  284. case tcell.KeyEscape:
  285. messageInputField.SetText("")
  286. messageInputField.SetTitle("")
  287. selectedMessage = -1
  288. messagesView.Highlight()
  289. return nil
  290. }
  291. return e
  292. }
  293. func newLoginForm(onLoginFormLoginButtonSelected func(), mfa bool) *tview.Form {
  294. w := tview.NewForm()
  295. w.
  296. AddButton("Login", onLoginFormLoginButtonSelected).
  297. SetButtonsAlign(tview.AlignCenter).
  298. SetBorder(true).
  299. SetBorderPadding(0, 0, 1, 0)
  300. if mfa {
  301. w.AddPasswordField("Code", "", 0, 0, nil)
  302. } else {
  303. w.
  304. AddInputField("Email", "", 0, nil, nil).
  305. AddPasswordField("Password", "", 0, 0, nil)
  306. }
  307. return w
  308. }
  309. // createChannelNode builds (encorporates unread channels in bold tag, otherwise dim, etc.) and returns a node according to the type of the given channel *c*.
  310. func createChannelNode(s *discordgo.State, c *discordgo.Channel) *tview.TreeNode {
  311. var cn *tview.TreeNode
  312. switch c.Type {
  313. case discordgo.ChannelTypeGuildText, discordgo.ChannelTypeGuildNews:
  314. tag := "[::d]"
  315. if channelIsUnread(s, c) {
  316. tag = "[::b]"
  317. }
  318. cn = tview.NewTreeNode(tag + util.ChannelToString(c) + "[::-]").
  319. SetReference(c.ID)
  320. case discordgo.ChannelTypeGuildCategory:
  321. cn = tview.NewTreeNode(c.Name).
  322. SetReference(c.ID)
  323. }
  324. return cn
  325. }
  326. // hasPermission returns a boolean that indicates whether the client user has the given permission *p* in the given channel ID *cID*.
  327. func hasPermission(s *discordgo.State, cID string, p int64) bool {
  328. perm, err := s.UserChannelPermissions(s.User.ID, cID)
  329. if err != nil {
  330. return false
  331. }
  332. return perm&p == p
  333. }
  334. // hasKeybinding returns a boolean that indicates whether the given keybinding string representation *k* is in the slice *ks*.
  335. func hasKeybinding(ks []string, k string) bool {
  336. for _, repr := range ks {
  337. if repr == k {
  338. return true
  339. }
  340. }
  341. return false
  342. }