message_input.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. package ui
  2. import (
  3. "io"
  4. "os"
  5. "os/exec"
  6. "strings"
  7. "github.com/atotto/clipboard"
  8. "github.com/diamondburned/arikawa/v3/api"
  9. "github.com/diamondburned/arikawa/v3/discord"
  10. "github.com/diamondburned/arikawa/v3/utils/json/option"
  11. "github.com/gdamore/tcell/v2"
  12. "github.com/rivo/tview"
  13. lua "github.com/yuin/gopher-lua"
  14. luar "layeh.com/gopher-luar"
  15. )
  16. type MessageInput struct {
  17. *tview.InputField
  18. core *Core
  19. }
  20. func NewMessageInput(c *Core) *MessageInput {
  21. mi := &MessageInput{
  22. InputField: tview.NewInputField(),
  23. core: c,
  24. }
  25. mi.SetFieldBackgroundColor(tview.Styles.PrimitiveBackgroundColor)
  26. mi.SetPlaceholder("Message...")
  27. mi.SetPlaceholderStyle(tcell.StyleDefault.Background(tview.Styles.PrimitiveBackgroundColor))
  28. mi.SetInputCapture(mi.onInputCapture)
  29. mi.SetTitleAlign(tview.AlignLeft)
  30. mi.SetBorder(true)
  31. mi.SetBorderPadding(0, 0, 1, 1)
  32. return mi
  33. }
  34. func (mi *MessageInput) onInputCapture(e *tcell.EventKey) *tcell.EventKey {
  35. keysTable, ok := mi.core.Config.State.GetGlobal("keys").(*lua.LTable)
  36. if !ok {
  37. return e
  38. }
  39. messageInputTable, ok := keysTable.RawGetString("messageInput").(*lua.LTable)
  40. if !ok {
  41. return e
  42. }
  43. var fn lua.LValue
  44. messageInputTable.ForEach(func(k, v lua.LValue) {
  45. keyTable := v.(*lua.LTable)
  46. if e.Name() == lua.LVAsString(keyTable.RawGetString("name")) {
  47. fn = keyTable.RawGetString("action")
  48. }
  49. })
  50. if fn != nil {
  51. mi.core.Config.State.CallByParam(lua.P{
  52. Fn: fn,
  53. NRet: 1,
  54. Protect: true,
  55. }, luar.New(mi.core.Config.State, mi.core), luar.New(mi.core.Config.State, e))
  56. // Returned value
  57. ret, ok := mi.core.Config.State.Get(-1).(*lua.LUserData)
  58. if !ok {
  59. return e
  60. }
  61. // Remove returned value
  62. mi.core.Config.State.Pop(1)
  63. ev, ok := ret.Value.(*tcell.EventKey)
  64. if ok {
  65. return ev
  66. }
  67. }
  68. // Defaults
  69. switch e.Name() {
  70. case "Enter":
  71. return mi.sendMessage()
  72. case "Esc":
  73. mi.
  74. SetText("").
  75. SetTitle("")
  76. mi.core.MessagesPanel.SelectedMessage = -1
  77. mi.core.MessagesPanel.Highlight()
  78. return nil
  79. }
  80. return e
  81. }
  82. func (mi *MessageInput) sendMessage() *tcell.EventKey {
  83. if mi.core.ChannelsTree.SelectedChannel == nil {
  84. return nil
  85. }
  86. t := strings.TrimSpace(mi.GetText())
  87. if t == "" {
  88. return nil
  89. }
  90. messagesLimit := mi.core.Config.State.GetGlobal("messagesLimit")
  91. ms, err := mi.core.State.Messages(mi.core.ChannelsTree.SelectedChannel.ID, uint(lua.LVAsNumber(messagesLimit)))
  92. if err != nil {
  93. return nil
  94. }
  95. if len(mi.core.MessagesPanel.GetHighlights()) != 0 {
  96. mID, err := discord.ParseSnowflake(mi.core.MessagesPanel.GetHighlights()[0])
  97. if err != nil {
  98. return nil
  99. }
  100. _, m := findMessageByID(ms, discord.MessageID(mID))
  101. d := api.SendMessageData{
  102. Content: t,
  103. Reference: m.Reference,
  104. AllowedMentions: &api.AllowedMentions{RepliedUser: option.False},
  105. }
  106. // If the title of the message InputField widget has "[@]" as a prefix, send the message as a reply and mention the replied user.
  107. if strings.HasPrefix(mi.GetTitle(), "[@]") {
  108. d.AllowedMentions.RepliedUser = option.True
  109. }
  110. go mi.core.State.SendMessageComplex(m.ChannelID, d)
  111. mi.core.MessagesPanel.SelectedMessage = -1
  112. mi.core.MessagesPanel.Highlight()
  113. mi.SetTitle("")
  114. } else {
  115. go mi.core.State.SendMessage(mi.core.ChannelsTree.SelectedChannel.ID, t)
  116. }
  117. mi.SetText("")
  118. return nil
  119. }
  120. func (mi *MessageInput) pasteClipboardContentLua(s *lua.LState) int {
  121. text, _ := clipboard.ReadAll()
  122. text = mi.GetText() + text
  123. mi.SetText(text)
  124. return returnNilLua(s)
  125. }
  126. func (mi *MessageInput) openExternalEditorLua(s *lua.LState) int {
  127. e := os.Getenv("EDITOR")
  128. if e == "" {
  129. return returnNilLua(s)
  130. }
  131. f, err := os.CreateTemp(os.TempDir(), "discordo-*.md")
  132. if err != nil {
  133. return returnNilLua(s)
  134. }
  135. defer os.Remove(f.Name())
  136. cmd := exec.Command(e, f.Name())
  137. cmd.Stdin = os.Stdin
  138. cmd.Stdout = os.Stdout
  139. mi.core.Application.Suspend(func() {
  140. err = cmd.Run()
  141. if err != nil {
  142. return
  143. }
  144. })
  145. b, err := io.ReadAll(f)
  146. if err != nil {
  147. return returnNilLua(s)
  148. }
  149. mi.SetText(string(b))
  150. return returnNilLua(s)
  151. }