2018-03-13 21:58:43 +02:00
|
|
|
// gomuks - A terminal Matrix client written in Go.
|
2019-01-17 14:13:25 +02:00
|
|
|
// Copyright (C) 2019 Tulir Asokan
|
2018-03-13 21:58:43 +02:00
|
|
|
//
|
|
|
|
// This program is free software: you can redistribute it and/or modify
|
2019-01-17 14:13:25 +02:00
|
|
|
// it under the terms of the GNU Affero General Public License as published by
|
2018-03-13 21:58:43 +02:00
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
//
|
|
|
|
// This program is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2019-01-17 14:13:25 +02:00
|
|
|
// GNU Affero General Public License for more details.
|
2018-03-13 21:58:43 +02:00
|
|
|
//
|
2019-01-17 14:13:25 +02:00
|
|
|
// You should have received a copy of the GNU Affero General Public License
|
|
|
|
// along with this program. If not, see <https://www.gnu.org/licenses/>.
|
2018-03-13 21:58:43 +02:00
|
|
|
|
2018-03-18 21:24:03 +02:00
|
|
|
package ui
|
2018-03-13 21:58:43 +02:00
|
|
|
|
|
|
|
import (
|
2019-01-17 14:13:25 +02:00
|
|
|
"bufio"
|
2018-03-17 14:32:01 +02:00
|
|
|
"fmt"
|
2019-01-17 14:13:25 +02:00
|
|
|
"os"
|
2018-03-15 21:38:43 +02:00
|
|
|
"time"
|
2018-03-17 14:32:01 +02:00
|
|
|
"unicode"
|
2018-03-13 21:58:43 +02:00
|
|
|
|
2018-07-02 12:30:42 +05:30
|
|
|
"github.com/kyokomi/emoji"
|
|
|
|
|
2018-11-14 00:00:35 +02:00
|
|
|
"maunium.net/go/mautrix"
|
2019-03-26 00:37:35 +02:00
|
|
|
"maunium.net/go/mauview"
|
2019-01-17 14:13:25 +02:00
|
|
|
"maunium.net/go/tcell"
|
|
|
|
|
2018-03-18 21:24:03 +02:00
|
|
|
"maunium.net/go/gomuks/config"
|
2018-04-09 23:45:54 +03:00
|
|
|
"maunium.net/go/gomuks/debug"
|
2018-03-18 21:24:03 +02:00
|
|
|
"maunium.net/go/gomuks/interface"
|
2018-04-11 15:05:42 +03:00
|
|
|
"maunium.net/go/gomuks/lib/notification"
|
2018-03-26 17:22:47 +03:00
|
|
|
"maunium.net/go/gomuks/matrix/pushrules"
|
|
|
|
"maunium.net/go/gomuks/matrix/rooms"
|
2018-04-14 11:50:18 +03:00
|
|
|
"maunium.net/go/gomuks/ui/messages/parser"
|
2018-03-18 21:24:03 +02:00
|
|
|
"maunium.net/go/gomuks/ui/widget"
|
2018-03-13 21:58:43 +02:00
|
|
|
)
|
|
|
|
|
2018-03-15 20:53:04 +02:00
|
|
|
type MainView struct {
|
2019-03-26 00:37:35 +02:00
|
|
|
flex *mauview.Flex
|
2018-03-15 20:53:04 +02:00
|
|
|
|
2018-05-27 14:54:07 +03:00
|
|
|
roomList *RoomList
|
2019-03-26 00:37:35 +02:00
|
|
|
roomView *mauview.Box
|
|
|
|
currentRoom *RoomView
|
2018-05-27 14:54:07 +03:00
|
|
|
rooms map[string]*RoomView
|
|
|
|
cmdProcessor *CommandProcessor
|
2019-03-26 00:37:35 +02:00
|
|
|
focused mauview.Focusable
|
|
|
|
|
|
|
|
modal mauview.Component
|
2018-03-15 20:53:04 +02:00
|
|
|
|
2018-03-24 22:14:17 +02:00
|
|
|
lastFocusTime time.Time
|
|
|
|
|
2018-03-18 21:24:03 +02:00
|
|
|
matrix ifc.MatrixContainer
|
|
|
|
gmx ifc.Gomuks
|
|
|
|
config *config.Config
|
2018-03-15 20:53:04 +02:00
|
|
|
parent *GomuksUI
|
|
|
|
}
|
|
|
|
|
2019-03-26 00:37:35 +02:00
|
|
|
func (ui *GomuksUI) NewMainView() mauview.Component {
|
2018-03-15 21:28:21 +02:00
|
|
|
mainView := &MainView{
|
2019-03-26 00:37:35 +02:00
|
|
|
flex: mauview.NewFlex().SetDirection(mauview.FlexColumn),
|
|
|
|
roomView: mauview.NewBox(nil).SetBorder(false),
|
2018-04-09 23:45:54 +03:00
|
|
|
rooms: make(map[string]*RoomView),
|
2018-03-15 20:53:04 +02:00
|
|
|
|
2018-03-23 14:44:36 +02:00
|
|
|
matrix: ui.gmx.Matrix(),
|
2018-03-15 20:53:04 +02:00
|
|
|
gmx: ui.gmx,
|
2018-03-18 21:24:03 +02:00
|
|
|
config: ui.gmx.Config(),
|
2018-03-15 20:53:04 +02:00
|
|
|
parent: ui,
|
|
|
|
}
|
2019-03-26 00:37:35 +02:00
|
|
|
mainView.roomList = NewRoomList(mainView)
|
2018-05-27 14:54:07 +03:00
|
|
|
mainView.cmdProcessor = NewCommandProcessor(mainView)
|
2018-03-15 20:53:04 +02:00
|
|
|
|
2019-03-26 00:37:35 +02:00
|
|
|
mainView.flex.
|
|
|
|
AddFixedComponent(mainView.roomList, 25).
|
|
|
|
AddFixedComponent(widget.NewBorder(), 1).
|
|
|
|
AddProportionalComponent(mainView.roomView, 1)
|
2018-05-16 20:09:09 +03:00
|
|
|
mainView.BumpFocus(nil)
|
2018-03-13 21:58:43 +02:00
|
|
|
|
2018-03-15 21:28:21 +02:00
|
|
|
ui.mainView = mainView
|
|
|
|
|
|
|
|
return mainView
|
2018-03-13 21:58:43 +02:00
|
|
|
}
|
|
|
|
|
2019-03-26 00:37:35 +02:00
|
|
|
func (view *MainView) ShowModal(modal mauview.Component) {
|
|
|
|
view.modal = modal
|
|
|
|
var ok bool
|
|
|
|
view.focused, ok = modal.(mauview.Focusable)
|
|
|
|
if !ok {
|
|
|
|
view.focused = nil
|
2019-03-28 23:28:27 +02:00
|
|
|
} else {
|
|
|
|
view.focused.Focus()
|
2019-03-26 00:37:35 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *MainView) HideModal() {
|
|
|
|
view.modal = nil
|
|
|
|
view.focused = view.roomView
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *MainView) Draw(screen mauview.Screen) {
|
2018-06-02 00:43:56 +03:00
|
|
|
if view.config.Preferences.HideRoomList {
|
2018-05-22 22:06:48 +03:00
|
|
|
view.roomView.Draw(screen)
|
|
|
|
} else {
|
2019-03-26 00:37:35 +02:00
|
|
|
view.flex.Draw(screen)
|
|
|
|
}
|
|
|
|
|
|
|
|
if view.modal != nil {
|
|
|
|
view.modal.Draw(screen)
|
2018-05-22 22:06:48 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-05-16 20:09:09 +03:00
|
|
|
func (view *MainView) BumpFocus(roomView *RoomView) {
|
2019-03-26 00:37:35 +02:00
|
|
|
if roomView != nil {
|
|
|
|
view.lastFocusTime = time.Now()
|
|
|
|
view.MarkRead(roomView)
|
|
|
|
}
|
2018-05-16 20:09:09 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
func (view *MainView) MarkRead(roomView *RoomView) {
|
|
|
|
if roomView != nil && roomView.Room.HasNewMessages() && roomView.MessageView().ScrollOffset == 0 {
|
|
|
|
msgList := roomView.MessageView().messages
|
|
|
|
msg := msgList[len(msgList)-1]
|
|
|
|
roomView.Room.MarkRead(msg.ID())
|
|
|
|
view.matrix.MarkRead(roomView.Room.ID, msg.ID())
|
|
|
|
}
|
2018-03-24 22:14:17 +02:00
|
|
|
}
|
|
|
|
|
2018-04-09 23:45:54 +03:00
|
|
|
func (view *MainView) InputChanged(roomView *RoomView, text string) {
|
2018-06-30 22:26:40 +02:00
|
|
|
if !roomView.config.Preferences.DisableTypingNotifs {
|
|
|
|
if len(text) == 0 {
|
|
|
|
go view.matrix.SendTyping(roomView.Room.ID, false)
|
|
|
|
} else if text[0] != '/' {
|
|
|
|
go view.matrix.SendTyping(roomView.Room.ID, true)
|
|
|
|
}
|
2018-03-15 21:28:21 +02:00
|
|
|
}
|
2018-03-15 20:53:04 +02:00
|
|
|
}
|
|
|
|
|
2018-03-17 14:32:01 +02:00
|
|
|
func findWordToTabComplete(text string) string {
|
|
|
|
output := ""
|
|
|
|
runes := []rune(text)
|
|
|
|
for i := len(runes) - 1; i >= 0; i-- {
|
|
|
|
if unicode.IsSpace(runes[i]) {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
output = string(runes[i]) + output
|
|
|
|
}
|
|
|
|
return output
|
|
|
|
}
|
|
|
|
|
2018-04-09 23:45:54 +03:00
|
|
|
func (view *MainView) InputSubmit(roomView *RoomView, text string) {
|
2018-03-20 19:14:39 +02:00
|
|
|
if len(text) == 0 {
|
|
|
|
return
|
|
|
|
} else if text[0] == '/' {
|
2018-05-27 14:54:07 +03:00
|
|
|
cmd := view.cmdProcessor.ParseCommand(roomView, text)
|
|
|
|
go view.cmdProcessor.HandleCommand(cmd)
|
2018-03-20 19:14:39 +02:00
|
|
|
} else {
|
2018-03-22 21:44:46 +02:00
|
|
|
view.SendMessage(roomView, text)
|
2018-03-20 19:14:39 +02:00
|
|
|
}
|
2018-03-22 16:44:24 +02:00
|
|
|
roomView.SetInputText("")
|
2018-03-20 19:14:39 +02:00
|
|
|
}
|
|
|
|
|
2018-04-09 23:45:54 +03:00
|
|
|
func (view *MainView) SendMessage(roomView *RoomView, text string) {
|
2018-03-22 21:44:46 +02:00
|
|
|
tempMessage := roomView.NewTempMessage("m.text", text)
|
2018-04-13 21:25:45 +03:00
|
|
|
go view.sendTempMessage(roomView, tempMessage, text)
|
2018-03-22 21:44:46 +02:00
|
|
|
}
|
|
|
|
|
2018-04-13 21:25:45 +03:00
|
|
|
func (view *MainView) sendTempMessage(roomView *RoomView, tempMessage ifc.Message, text string) {
|
2018-04-18 18:35:24 +03:00
|
|
|
defer debug.Recover()
|
2018-05-10 15:47:24 +03:00
|
|
|
debug.Print("Sending message", tempMessage.Type(), text)
|
2018-07-02 12:30:42 +05:30
|
|
|
if !roomView.config.Preferences.DisableEmojis {
|
|
|
|
text = emoji.Sprint(text)
|
|
|
|
}
|
2018-04-18 14:20:57 +03:00
|
|
|
eventID, err := view.matrix.SendMarkdownMessage(roomView.Room.ID, tempMessage.Type(), text)
|
2018-03-22 21:44:46 +02:00
|
|
|
if err != nil {
|
2018-04-09 23:45:54 +03:00
|
|
|
tempMessage.SetState(ifc.MessageStateFailed)
|
2018-11-14 00:00:35 +02:00
|
|
|
if httpErr, ok := err.(mautrix.HTTPError); ok {
|
2019-03-26 19:57:44 +02:00
|
|
|
if respErr := httpErr.RespError; respErr != nil {
|
2018-04-30 22:35:52 +03:00
|
|
|
// Show shorter version if available
|
|
|
|
err = respErr
|
|
|
|
}
|
|
|
|
}
|
2018-04-21 19:41:19 +03:00
|
|
|
roomView.AddServiceMessage(fmt.Sprintf("Failed to send message: %v", err))
|
2018-04-30 22:35:52 +03:00
|
|
|
view.parent.Render()
|
2018-03-22 21:44:46 +02:00
|
|
|
} else {
|
2018-05-10 15:47:24 +03:00
|
|
|
debug.Print("Event ID received:", eventID)
|
2018-03-22 21:44:46 +02:00
|
|
|
roomView.MessageView().UpdateMessageID(tempMessage, eventID)
|
|
|
|
}
|
2018-03-15 20:53:04 +02:00
|
|
|
}
|
|
|
|
|
2018-05-23 00:44:29 +03:00
|
|
|
func (view *MainView) ShowBare(roomView *RoomView) {
|
2019-03-26 00:37:35 +02:00
|
|
|
if roomView == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
_, height := view.parent.app.Screen().Size()
|
2018-05-23 00:44:29 +03:00
|
|
|
view.parent.app.Suspend(func() {
|
|
|
|
print("\033[2J\033[0;0H")
|
|
|
|
// We don't know how much space there exactly is. Too few messages looks weird,
|
|
|
|
// and too many messages shouldn't cause any problems, so we just show too many.
|
|
|
|
height *= 2
|
|
|
|
fmt.Println(roomView.MessageView().CapturePlaintext(height))
|
|
|
|
fmt.Println("Press enter to return to normal mode.")
|
|
|
|
reader := bufio.NewReader(os.Stdin)
|
|
|
|
reader.ReadRune()
|
|
|
|
print("\033[2J\033[0;0H")
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2019-03-26 00:37:35 +02:00
|
|
|
func (view *MainView) OnKeyEvent(event mauview.KeyEvent) bool {
|
|
|
|
view.BumpFocus(view.currentRoom)
|
|
|
|
|
|
|
|
if view.modal != nil {
|
|
|
|
return view.modal.OnKeyEvent(event)
|
|
|
|
}
|
2018-03-24 22:14:17 +02:00
|
|
|
|
2019-03-26 00:37:35 +02:00
|
|
|
k := event.Key()
|
|
|
|
c := event.Rune()
|
|
|
|
if event.Modifiers() == tcell.ModCtrl || event.Modifiers() == tcell.ModAlt {
|
2018-05-22 22:06:48 +03:00
|
|
|
switch {
|
|
|
|
case k == tcell.KeyDown:
|
2018-04-23 22:49:17 +03:00
|
|
|
view.SwitchRoom(view.roomList.Next())
|
2018-05-22 22:06:48 +03:00
|
|
|
case k == tcell.KeyUp:
|
2018-04-23 22:49:17 +03:00
|
|
|
view.SwitchRoom(view.roomList.Previous())
|
2018-05-22 22:06:48 +03:00
|
|
|
case k == tcell.KeyEnter:
|
2019-03-26 00:37:35 +02:00
|
|
|
view.ShowModal(NewFuzzySearchModal(view, 42, 12))
|
|
|
|
case k == tcell.KeyHome:
|
|
|
|
msgView := view.currentRoom.MessageView()
|
|
|
|
msgView.AddScrollOffset(msgView.TotalHeight())
|
|
|
|
case k == tcell.KeyEnd:
|
|
|
|
msgView := view.currentRoom.MessageView()
|
|
|
|
msgView.AddScrollOffset(-msgView.TotalHeight())
|
|
|
|
case k == tcell.KeyCtrlN:
|
|
|
|
return view.flex.OnKeyEvent(tcell.NewEventKey(tcell.KeyEnter, '\n', event.Modifiers()))
|
2018-10-18 17:02:38 +03:00
|
|
|
case c == 'a':
|
|
|
|
view.SwitchRoom(view.roomList.NextWithActivity())
|
2018-05-22 22:06:48 +03:00
|
|
|
case c == 'l':
|
2019-03-26 00:37:35 +02:00
|
|
|
view.ShowBare(view.currentRoom)
|
2018-03-25 14:21:59 +03:00
|
|
|
default:
|
2019-03-26 00:37:35 +02:00
|
|
|
goto defaultHandler
|
2018-03-13 21:58:43 +02:00
|
|
|
}
|
2019-03-26 00:37:35 +02:00
|
|
|
return true
|
2018-06-30 22:28:53 +02:00
|
|
|
} else if k == tcell.KeyAltDown || k == tcell.KeyCtrlDown {
|
|
|
|
view.SwitchRoom(view.roomList.Next())
|
2019-03-26 00:37:35 +02:00
|
|
|
return true
|
2018-06-30 22:28:53 +02:00
|
|
|
} else if k == tcell.KeyAltUp || k == tcell.KeyCtrlUp {
|
|
|
|
view.SwitchRoom(view.roomList.Previous())
|
2019-03-26 00:37:35 +02:00
|
|
|
return true
|
|
|
|
} else if view.currentRoom != nil &&
|
|
|
|
(k == tcell.KeyPgUp || k == tcell.KeyPgDn ||
|
|
|
|
k == tcell.KeyUp || k == tcell.KeyDown ||
|
|
|
|
k == tcell.KeyEnd || k == tcell.KeyHome) {
|
|
|
|
// TODO these should be in the RoomView key handler
|
|
|
|
msgView := view.currentRoom.MessageView()
|
2018-03-25 12:35:50 +03:00
|
|
|
|
|
|
|
if msgView.IsAtTop() && (k == tcell.KeyPgUp || k == tcell.KeyUp) {
|
2019-03-26 00:37:35 +02:00
|
|
|
go view.LoadHistory(view.currentRoom.Room.ID)
|
2018-03-25 12:35:50 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
switch k {
|
|
|
|
case tcell.KeyPgUp:
|
|
|
|
msgView.AddScrollOffset(msgView.Height() / 2)
|
|
|
|
case tcell.KeyPgDn:
|
|
|
|
msgView.AddScrollOffset(-msgView.Height() / 2)
|
2019-03-26 00:37:35 +02:00
|
|
|
default:
|
|
|
|
goto defaultHandler
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
2019-03-26 00:37:35 +02:00
|
|
|
return true
|
|
|
|
} else if k == tcell.KeyEnter {
|
|
|
|
view.InputSubmit(view.currentRoom, view.currentRoom.input.GetText())
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
defaultHandler:
|
|
|
|
if view.config.Preferences.HideRoomList {
|
|
|
|
debug.Print("Key event going to default handler (direct to roomview)", event)
|
|
|
|
return view.roomView.OnKeyEvent(event)
|
2018-03-13 21:58:43 +02:00
|
|
|
}
|
2019-03-26 00:37:35 +02:00
|
|
|
debug.Print("Key event going to default handler (flex)", event)
|
|
|
|
return view.flex.OnKeyEvent(event)
|
2018-03-13 21:58:43 +02:00
|
|
|
}
|
|
|
|
|
2018-03-25 12:35:50 +03:00
|
|
|
const WheelScrollOffsetDiff = 3
|
|
|
|
|
2019-03-26 00:37:35 +02:00
|
|
|
func (view *MainView) OnMouseEvent(event mauview.MouseEvent) bool {
|
2019-03-28 23:28:27 +02:00
|
|
|
if view.modal != nil {
|
|
|
|
return view.modal.OnMouseEvent(event)
|
|
|
|
}
|
2019-03-26 00:37:35 +02:00
|
|
|
if view.config.Preferences.HideRoomList {
|
|
|
|
return view.roomView.OnMouseEvent(event)
|
2018-03-25 12:35:50 +03:00
|
|
|
}
|
2019-03-26 00:37:35 +02:00
|
|
|
return view.flex.OnMouseEvent(event)
|
|
|
|
/*if event.Buttons() == tcell.ButtonNone || event.HasMotion() {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
|
|
|
view.BumpFocus(view.currentRoom)
|
2018-03-25 12:35:50 +03:00
|
|
|
|
|
|
|
x, y := event.Position()
|
|
|
|
|
2018-04-30 12:40:28 +03:00
|
|
|
switch {
|
2019-03-26 00:37:35 +02:00
|
|
|
case x >= 27:
|
|
|
|
view.roomView.OnMouseEvent(mauview.OffsetMouseEvent(event, -27, 0))
|
|
|
|
view.roomView.Focus()
|
|
|
|
view.focused = view.roomView
|
|
|
|
case x <= 25:
|
|
|
|
view.roomList.OnMouseEvent(event)
|
|
|
|
view.roomList.Focus()
|
|
|
|
view.focused = view.roomList
|
2018-04-30 12:40:28 +03:00
|
|
|
default:
|
|
|
|
debug.Print("Unhandled mouse event:", event.Buttons(), event.Modifiers(), x, y)
|
2018-03-24 22:14:17 +02:00
|
|
|
}
|
2019-03-26 00:37:35 +02:00
|
|
|
return false*/
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *MainView) OnPasteEvent(event mauview.PasteEvent) bool {
|
|
|
|
if view.modal != nil {
|
|
|
|
return view.modal.OnPasteEvent(event)
|
|
|
|
} else if view.config.Preferences.HideRoomList {
|
|
|
|
return view.roomView.OnPasteEvent(event)
|
|
|
|
}
|
|
|
|
return view.flex.OnPasteEvent(event)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *MainView) Focus() {
|
|
|
|
if view.focused != nil {
|
|
|
|
view.focused.Focus()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *MainView) Blur() {
|
|
|
|
if view.focused != nil {
|
|
|
|
view.focused.Blur()
|
|
|
|
}
|
2018-03-24 22:14:17 +02:00
|
|
|
}
|
|
|
|
|
2018-04-24 02:13:17 +03:00
|
|
|
func (view *MainView) SwitchRoom(tag string, room *rooms.Room) {
|
2018-04-23 23:22:18 +03:00
|
|
|
if room == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-04-23 22:49:17 +03:00
|
|
|
roomView := view.rooms[room.ID]
|
2018-05-16 20:09:09 +03:00
|
|
|
if roomView == nil {
|
|
|
|
debug.Print("Tried to switch to non-nil room with nil roomView!")
|
|
|
|
debug.Print(tag, room)
|
|
|
|
return
|
2018-03-26 17:22:47 +03:00
|
|
|
}
|
2019-03-26 00:37:35 +02:00
|
|
|
view.roomView.SetInnerComponent(roomView)
|
|
|
|
view.currentRoom = roomView
|
2018-05-16 20:09:09 +03:00
|
|
|
view.MarkRead(roomView)
|
2018-04-24 02:13:17 +03:00
|
|
|
view.roomList.SetSelected(tag, room)
|
2018-03-15 20:53:04 +02:00
|
|
|
view.parent.Render()
|
|
|
|
}
|
|
|
|
|
2018-03-22 19:51:20 +02:00
|
|
|
func (view *MainView) SaveAllHistory() {
|
|
|
|
for _, room := range view.rooms {
|
|
|
|
err := room.SaveHistory(view.config.HistoryDir)
|
|
|
|
if err != nil {
|
|
|
|
debug.Printf("Failed to save history of %s: %v", room.Room.GetTitle(), err)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-23 22:49:17 +03:00
|
|
|
func (view *MainView) addRoomPage(room *rooms.Room) {
|
2019-03-26 00:37:35 +02:00
|
|
|
if _, ok := view.rooms[room.ID]; !ok {
|
2018-04-23 22:49:17 +03:00
|
|
|
roomView := NewRoomView(view, room).
|
2018-03-22 16:44:24 +02:00
|
|
|
SetInputSubmitFunc(view.InputSubmit).
|
2019-03-26 00:37:35 +02:00
|
|
|
SetInputChangedFunc(view.InputChanged)
|
2018-04-23 22:49:17 +03:00
|
|
|
view.rooms[room.ID] = roomView
|
2018-03-16 16:24:11 +02:00
|
|
|
roomView.UpdateUserList()
|
2018-03-22 19:51:20 +02:00
|
|
|
|
2018-04-24 22:08:09 +03:00
|
|
|
_, err := roomView.LoadHistory(view.matrix, view.config.HistoryDir)
|
2018-03-22 19:51:20 +02:00
|
|
|
if err != nil {
|
|
|
|
debug.Printf("Failed to load history of %s: %v", roomView.Room.GetTitle(), err)
|
|
|
|
}
|
2018-03-16 16:24:11 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-23 22:49:17 +03:00
|
|
|
func (view *MainView) GetRoom(roomID string) ifc.RoomView {
|
2018-04-24 02:13:17 +03:00
|
|
|
room, ok := view.rooms[roomID]
|
|
|
|
if !ok {
|
2018-05-02 14:17:06 +03:00
|
|
|
view.AddRoom(view.matrix.GetRoom(roomID))
|
2018-04-30 12:40:28 +03:00
|
|
|
room, ok := view.rooms[roomID]
|
|
|
|
if !ok {
|
|
|
|
return nil
|
|
|
|
}
|
2018-04-24 02:13:17 +03:00
|
|
|
return room
|
|
|
|
}
|
|
|
|
return room
|
2018-03-18 17:34:42 +02:00
|
|
|
}
|
|
|
|
|
2018-04-30 23:12:43 +03:00
|
|
|
func (view *MainView) AddRoom(room *rooms.Room) {
|
|
|
|
if view.roomList.Contains(room.ID) {
|
2018-05-16 20:09:09 +03:00
|
|
|
debug.Print("Add aborted (room exists)", room.ID, room.GetTitle())
|
2018-03-16 16:24:11 +02:00
|
|
|
return
|
|
|
|
}
|
2018-05-16 20:09:09 +03:00
|
|
|
debug.Print("Adding", room.ID, room.GetTitle())
|
2018-04-23 22:49:17 +03:00
|
|
|
view.roomList.Add(room)
|
|
|
|
view.addRoomPage(room)
|
2018-04-24 22:46:16 +03:00
|
|
|
if !view.roomList.HasSelected() {
|
|
|
|
view.SwitchRoom(view.roomList.First())
|
|
|
|
}
|
2018-03-16 16:24:11 +02:00
|
|
|
}
|
|
|
|
|
2018-04-30 23:12:43 +03:00
|
|
|
func (view *MainView) RemoveRoom(room *rooms.Room) {
|
|
|
|
roomView := view.GetRoom(room.ID)
|
2018-03-25 14:21:59 +03:00
|
|
|
if roomView == nil {
|
2018-05-16 20:09:09 +03:00
|
|
|
debug.Print("Remove aborted (not found)", room.ID, room.GetTitle())
|
2018-03-16 16:24:11 +02:00
|
|
|
return
|
|
|
|
}
|
2018-05-16 20:09:09 +03:00
|
|
|
debug.Print("Removing", room.ID, room.GetTitle())
|
2018-04-23 22:49:17 +03:00
|
|
|
|
2018-04-30 23:12:43 +03:00
|
|
|
view.roomList.Remove(room)
|
2018-04-23 22:49:17 +03:00
|
|
|
view.SwitchRoom(view.roomList.Selected())
|
|
|
|
|
2018-04-30 23:12:43 +03:00
|
|
|
delete(view.rooms, room.ID)
|
2018-04-23 22:49:17 +03:00
|
|
|
|
2018-03-20 12:16:32 +02:00
|
|
|
view.parent.Render()
|
2018-03-16 16:24:11 +02:00
|
|
|
}
|
|
|
|
|
2018-04-24 16:51:40 +03:00
|
|
|
func (view *MainView) SetRooms(rooms map[string]*rooms.Room) {
|
2018-03-15 20:53:04 +02:00
|
|
|
view.roomList.Clear()
|
2018-04-09 23:45:54 +03:00
|
|
|
view.rooms = make(map[string]*RoomView)
|
2018-04-24 16:51:40 +03:00
|
|
|
for _, room := range rooms {
|
2018-04-30 22:28:29 +03:00
|
|
|
if room.HasLeft {
|
|
|
|
continue
|
|
|
|
}
|
2018-04-23 22:49:17 +03:00
|
|
|
view.roomList.Add(room)
|
|
|
|
view.addRoomPage(room)
|
2018-04-24 02:13:17 +03:00
|
|
|
}
|
|
|
|
view.SwitchRoom(view.roomList.First())
|
|
|
|
}
|
|
|
|
|
2018-05-16 20:09:09 +03:00
|
|
|
func (view *MainView) UpdateTags(room *rooms.Room) {
|
|
|
|
if !view.roomList.Contains(room.ID) {
|
|
|
|
return
|
2018-03-13 21:58:43 +02:00
|
|
|
}
|
2018-05-16 20:09:09 +03:00
|
|
|
view.roomList.Remove(room)
|
|
|
|
view.roomList.Add(room)
|
2018-03-13 21:58:43 +02:00
|
|
|
}
|
|
|
|
|
2018-03-15 20:53:04 +02:00
|
|
|
func (view *MainView) SetTyping(room string, users []string) {
|
|
|
|
roomView, ok := view.rooms[room]
|
2018-03-15 00:14:39 +02:00
|
|
|
if ok {
|
2018-03-15 19:45:52 +02:00
|
|
|
roomView.SetTyping(users)
|
2018-03-15 20:53:04 +02:00
|
|
|
view.parent.Render()
|
2018-03-15 00:14:39 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-26 18:04:10 +03:00
|
|
|
func sendNotification(room *rooms.Room, sender, text string, critical, sound bool) {
|
2018-03-26 17:22:47 +03:00
|
|
|
if room.GetTitle() != sender {
|
|
|
|
sender = fmt.Sprintf("%s (%s)", sender, room.GetTitle())
|
|
|
|
}
|
2018-04-25 07:01:00 +03:00
|
|
|
debug.Printf("Sending notification with body \"%s\" from %s in room ID %s (critical=%v, sound=%v)", text, sender, room.ID, critical, sound)
|
2018-03-26 18:04:10 +03:00
|
|
|
notification.Send(sender, text, critical, sound)
|
2018-03-26 17:22:47 +03:00
|
|
|
}
|
|
|
|
|
2018-04-09 23:45:54 +03:00
|
|
|
func (view *MainView) NotifyMessage(room *rooms.Room, message ifc.Message, should pushrules.PushActionArrayShould) {
|
2018-05-16 21:42:07 +03:00
|
|
|
view.roomList.Bump(room)
|
2018-05-17 16:29:15 +03:00
|
|
|
if message.SenderID() == view.config.UserID {
|
2018-05-16 21:42:07 +03:00
|
|
|
return
|
|
|
|
}
|
2018-03-26 20:55:52 +03:00
|
|
|
// Whether or not the room where the message came is the currently shown room.
|
2018-04-24 02:13:17 +03:00
|
|
|
isCurrent := room == view.roomList.SelectedRoom()
|
2018-03-26 20:55:52 +03:00
|
|
|
// Whether or not the terminal window is focused.
|
2018-05-16 20:09:09 +03:00
|
|
|
recentlyFocused := time.Now().Add(-30 * time.Second).Before(view.lastFocusTime)
|
|
|
|
isFocused := time.Now().Add(-5 * time.Second).Before(view.lastFocusTime)
|
2018-03-26 20:55:52 +03:00
|
|
|
|
|
|
|
// Whether or not the push rules say this message should be notified about.
|
2018-05-16 21:42:07 +03:00
|
|
|
shouldNotify := should.Notify || !should.NotifySpecified
|
2018-03-26 20:55:52 +03:00
|
|
|
|
2018-05-16 20:09:09 +03:00
|
|
|
if !isCurrent || !isFocused {
|
2018-03-26 20:55:52 +03:00
|
|
|
// The message is not in the current room, show new message status in room list.
|
2018-05-16 20:09:09 +03:00
|
|
|
room.AddUnread(message.ID(), shouldNotify, should.Highlight)
|
|
|
|
} else {
|
|
|
|
view.matrix.MarkRead(room.ID, message.ID())
|
2018-03-26 17:22:47 +03:00
|
|
|
}
|
2018-03-26 20:55:52 +03:00
|
|
|
|
2018-05-16 20:09:09 +03:00
|
|
|
if shouldNotify && !recentlyFocused {
|
2018-03-26 20:55:52 +03:00
|
|
|
// Push rules say notify and the terminal is not focused, send desktop notification.
|
2018-03-26 18:04:10 +03:00
|
|
|
shouldPlaySound := should.PlaySound && should.SoundName == "default"
|
2018-04-13 21:25:45 +03:00
|
|
|
sendNotification(room, message.Sender(), message.NotificationContent(), should.Highlight, shouldPlaySound)
|
2018-03-26 17:22:47 +03:00
|
|
|
}
|
2018-03-26 20:55:52 +03:00
|
|
|
|
2018-04-09 23:45:54 +03:00
|
|
|
message.SetIsHighlight(should.Highlight)
|
2018-03-13 21:58:43 +02:00
|
|
|
}
|
2018-03-17 15:48:31 +02:00
|
|
|
|
2018-05-15 18:45:09 +03:00
|
|
|
func (view *MainView) InitialSyncDone() {
|
|
|
|
view.roomList.Clear()
|
|
|
|
for _, room := range view.rooms {
|
|
|
|
view.roomList.Add(room.Room)
|
2018-05-16 14:28:57 +03:00
|
|
|
room.UpdateUserList()
|
2018-05-15 18:45:09 +03:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-24 22:08:09 +03:00
|
|
|
func (view *MainView) LoadHistory(room string) {
|
2018-04-18 18:35:24 +03:00
|
|
|
defer debug.Recover()
|
2018-03-18 17:34:42 +02:00
|
|
|
roomView := view.rooms[room]
|
2018-03-20 12:16:32 +02:00
|
|
|
|
|
|
|
batch := roomView.Room.PrevBatch
|
|
|
|
lockTime := time.Now().Unix() + 1
|
|
|
|
|
2018-03-22 17:36:06 +02:00
|
|
|
roomView.Room.LockHistory()
|
2018-03-20 12:16:32 +02:00
|
|
|
roomView.MessageView().LoadingMessages = true
|
|
|
|
defer func() {
|
2018-03-22 17:36:06 +02:00
|
|
|
roomView.Room.UnlockHistory()
|
2018-03-20 12:16:32 +02:00
|
|
|
roomView.MessageView().LoadingMessages = false
|
|
|
|
}()
|
|
|
|
|
|
|
|
// There's no clean way to try to lock a mutex, so we just check if we still
|
|
|
|
// want to continue after we get the lock. This function should always be ran
|
|
|
|
// in a goroutine, so the blocking doesn't matter.
|
|
|
|
if time.Now().Unix() >= lockTime || batch != roomView.Room.PrevBatch {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-04-24 22:08:09 +03:00
|
|
|
debug.Print("Fetching history for", room, "starting from", batch)
|
2018-03-20 12:16:32 +02:00
|
|
|
history, prevBatch, err := view.matrix.GetHistory(roomView.Room.ID, batch, 50)
|
2018-03-18 17:34:42 +02:00
|
|
|
if err != nil {
|
2018-04-09 23:45:54 +03:00
|
|
|
roomView.AddServiceMessage("Failed to fetch history")
|
2018-03-18 21:24:03 +02:00
|
|
|
debug.Print("Failed to fetch history for", roomView.Room.ID, err)
|
2018-03-18 17:34:42 +02:00
|
|
|
return
|
|
|
|
}
|
2018-03-20 12:16:32 +02:00
|
|
|
roomView.Room.PrevBatch = prevBatch
|
2018-03-18 17:34:42 +02:00
|
|
|
for _, evt := range history {
|
2018-09-05 10:55:48 +03:00
|
|
|
message := view.ParseEvent(roomView, evt)
|
2018-03-23 01:00:13 +02:00
|
|
|
if message != nil {
|
2018-04-09 23:45:54 +03:00
|
|
|
roomView.AddMessage(message, ifc.PrependMessage)
|
2018-03-18 17:34:42 +02:00
|
|
|
}
|
|
|
|
}
|
2018-03-22 19:51:20 +02:00
|
|
|
err = roomView.SaveHistory(view.config.HistoryDir)
|
|
|
|
if err != nil {
|
2018-03-23 01:07:44 +02:00
|
|
|
debug.Printf("Failed to save history of %s: %v", roomView.Room.GetTitle(), err)
|
2018-03-22 19:51:20 +02:00
|
|
|
}
|
2018-05-17 16:29:15 +03:00
|
|
|
view.config.PutRoom(roomView.Room)
|
2018-03-20 12:16:32 +02:00
|
|
|
view.parent.Render()
|
2018-03-18 17:34:42 +02:00
|
|
|
}
|
|
|
|
|
2018-11-14 00:00:35 +02:00
|
|
|
func (view *MainView) ParseEvent(roomView ifc.RoomView, evt *mautrix.Event) ifc.Message {
|
2018-04-18 18:35:24 +03:00
|
|
|
return parser.ParseEvent(view.matrix, roomView.MxRoom(), evt)
|
2018-03-18 17:34:42 +02:00
|
|
|
}
|