message_input.go 3.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147
  1. package cmd
  2. import (
  3. "log"
  4. "os"
  5. "os/exec"
  6. "strings"
  7. "github.com/atotto/clipboard"
  8. "github.com/ayn2op/discordo/internal/constants"
  9. "github.com/diamondburned/arikawa/v3/api"
  10. "github.com/diamondburned/arikawa/v3/discord"
  11. "github.com/diamondburned/arikawa/v3/utils/json/option"
  12. "github.com/gdamore/tcell/v2"
  13. "github.com/rivo/tview"
  14. )
  15. type MessageInput struct {
  16. *tview.TextArea
  17. replyMessageIdx int
  18. }
  19. func newMessageInput() *MessageInput {
  20. mi := &MessageInput{
  21. TextArea: tview.NewTextArea(),
  22. replyMessageIdx: -1,
  23. }
  24. mi.SetTextStyle(tcell.StyleDefault.Background(tcell.GetColor(cfg.Theme.BackgroundColor)))
  25. mi.SetClipboard(func(s string) {
  26. _ = clipboard.WriteAll(s)
  27. }, func() string {
  28. text, _ := clipboard.ReadAll()
  29. return text
  30. })
  31. mi.SetInputCapture(mi.onInputCapture)
  32. mi.SetBackgroundColor(tcell.GetColor(cfg.Theme.BackgroundColor))
  33. mi.SetTitleColor(tcell.GetColor(cfg.Theme.TitleColor))
  34. mi.SetTitleAlign(tview.AlignLeft)
  35. p := cfg.Theme.BorderPadding
  36. mi.SetBorder(cfg.Theme.Border)
  37. mi.SetBorderColor(tcell.GetColor(cfg.Theme.BorderColor))
  38. mi.SetBorderPadding(p[0], p[1], p[2], p[3])
  39. return mi
  40. }
  41. func (mi *MessageInput) reset() {
  42. mi.SetTitle("")
  43. mi.SetText("", true)
  44. }
  45. func (mi *MessageInput) onInputCapture(event *tcell.EventKey) *tcell.EventKey {
  46. switch event.Name() {
  47. case cfg.Keys.MessageInput.Send:
  48. mi.sendAction()
  49. return nil
  50. case "Alt+Enter":
  51. return tcell.NewEventKey(tcell.KeyEnter, 0, tcell.ModNone)
  52. case cfg.Keys.MessageInput.LaunchEditor:
  53. mainFlex.messageInput.launchEditorAction()
  54. return nil
  55. case cfg.Keys.Cancel:
  56. mi.replyMessageIdx = -1
  57. mi.reset()
  58. return nil
  59. }
  60. return event
  61. }
  62. func (mi *MessageInput) sendAction() {
  63. if !mainFlex.guildsTree.selectedChannelID.IsValid() {
  64. return
  65. }
  66. text := strings.TrimSpace(mi.GetText())
  67. if text == "" {
  68. return
  69. }
  70. if mi.replyMessageIdx != -1 {
  71. ms, err := discordState.Cabinet.Messages(mainFlex.guildsTree.selectedChannelID)
  72. if err != nil {
  73. log.Println(err)
  74. return
  75. }
  76. data := api.SendMessageData{
  77. Content: text,
  78. Reference: &discord.MessageReference{MessageID: ms[mi.replyMessageIdx].ID},
  79. AllowedMentions: &api.AllowedMentions{RepliedUser: option.False},
  80. }
  81. if strings.HasPrefix(mi.GetTitle(), "[@]") {
  82. data.AllowedMentions.RepliedUser = option.True
  83. }
  84. go discordState.SendMessageComplex(mainFlex.guildsTree.selectedChannelID, data)
  85. } else {
  86. go discordState.SendMessage(mainFlex.guildsTree.selectedChannelID, text)
  87. }
  88. mi.replyMessageIdx = -1
  89. mainFlex.messagesText.Highlight()
  90. mi.reset()
  91. }
  92. func (mi *MessageInput) launchEditorAction() {
  93. e := cfg.Editor
  94. if e == "default" {
  95. e = os.Getenv("EDITOR")
  96. }
  97. f, err := os.CreateTemp("", constants.TmpFilePattern)
  98. if err != nil {
  99. log.Println(err)
  100. return
  101. }
  102. f.Close()
  103. defer os.Remove(f.Name())
  104. cmd := exec.Command(e, f.Name())
  105. cmd.Stdin = os.Stdin
  106. cmd.Stdout = os.Stdout
  107. cmd.Stderr = os.Stderr
  108. app.Suspend(func() {
  109. err := cmd.Run()
  110. if err != nil {
  111. log.Println(err)
  112. return
  113. }
  114. })
  115. msg, err := os.ReadFile(f.Name())
  116. if err != nil {
  117. log.Println(err)
  118. return
  119. }
  120. mi.SetText(string(msg), true)
  121. }