// gomuks - A terminal Matrix client written in Go. // Copyright (C) 2018 Tulir Asokan // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // 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 // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . package ui import ( "fmt" "strings" "time" "unicode" "maunium.net/go/gomatrix" "maunium.net/go/gomuks/config" "maunium.net/go/gomuks/debug" "maunium.net/go/gomuks/interface" "maunium.net/go/gomuks/lib/notification" "maunium.net/go/gomuks/matrix/pushrules" "maunium.net/go/gomuks/matrix/rooms" "maunium.net/go/gomuks/ui/messages/parser" "maunium.net/go/gomuks/ui/widget" "maunium.net/go/tcell" "maunium.net/go/tview" ) type MainView struct { *tview.Flex roomList *RoomList roomView *tview.Pages rooms map[string]*RoomView lastFocusTime time.Time matrix ifc.MatrixContainer gmx ifc.Gomuks config *config.Config parent *GomuksUI } func (ui *GomuksUI) NewMainView() tview.Primitive { mainView := &MainView{ Flex: tview.NewFlex(), roomList: NewRoomList(), roomView: tview.NewPages(), rooms: make(map[string]*RoomView), matrix: ui.gmx.Matrix(), gmx: ui.gmx, config: ui.gmx.Config(), parent: ui, } mainView.SetDirection(tview.FlexColumn) mainView.AddItem(mainView.roomList, 25, 0, false) mainView.AddItem(widget.NewBorder(), 1, 0, false) mainView.AddItem(mainView.roomView, 0, 1, true) mainView.BumpFocus() ui.mainView = mainView return mainView } func (view *MainView) BumpFocus() { view.lastFocusTime = time.Now() } func (view *MainView) InputChanged(roomView *RoomView, text string) { if len(text) == 0 { go view.matrix.SendTyping(roomView.Room.ID, false) } else if text[0] != '/' { go view.matrix.SendTyping(roomView.Room.ID, true) } } 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 } func (view *MainView) InputSubmit(roomView *RoomView, text string) { if len(text) == 0 { return } else if text[0] == '/' { args := strings.SplitN(text, " ", 2) command := strings.ToLower(args[0]) args = args[1:] go view.HandleCommand(roomView, command, args) } else { view.SendMessage(roomView, text) } roomView.SetInputText("") } func (view *MainView) SendMessage(roomView *RoomView, text string) { tempMessage := roomView.NewTempMessage("m.text", text) go view.sendTempMessage(roomView, tempMessage, text) } func (view *MainView) sendTempMessage(roomView *RoomView, tempMessage ifc.Message, text string) { defer debug.Recover() eventID, err := view.matrix.SendMarkdownMessage(roomView.Room.ID, tempMessage.Type(), text) if err != nil { tempMessage.SetState(ifc.MessageStateFailed) if httpErr, ok := err.(gomatrix.HTTPError); ok { if respErr, ok := httpErr.WrappedError.(gomatrix.RespError); ok { // Show shorter version if available err = respErr } } roomView.AddServiceMessage(fmt.Sprintf("Failed to send message: %v", err)) view.parent.Render() } else { roomView.MessageView().UpdateMessageID(tempMessage, eventID) } } func (view *MainView) HandleCommand(roomView *RoomView, command string, args []string) { defer debug.Recover() debug.Print("Handling command", command, args) switch command { case "/me": text := strings.Join(args, " ") tempMessage := roomView.NewTempMessage("m.emote", text) go view.sendTempMessage(roomView, tempMessage, text) view.parent.Render() case "/quit": view.gmx.Stop() case "/clearcache": view.config.Clear() view.gmx.Stop() case "/panic": panic("This is a test panic.") case "/part", "/leave": debug.Print("Leave room result:", view.matrix.LeaveRoom(roomView.Room.ID)) case "/join": if len(args) == 0 { roomView.AddServiceMessage("Usage: /join ") break } debug.Print("Join room result:", view.matrix.JoinRoom(args[0])) default: roomView.AddServiceMessage("Unknown command.") } } func (view *MainView) KeyEventHandler(roomView *RoomView, key *tcell.EventKey) *tcell.EventKey { view.BumpFocus() k := key.Key() if key.Modifiers() == tcell.ModCtrl || key.Modifiers() == tcell.ModAlt { switch k { case tcell.KeyDown: view.SwitchRoom(view.roomList.Next()) case tcell.KeyUp: view.SwitchRoom(view.roomList.Previous()) default: return key } } else if k == tcell.KeyPgUp || k == tcell.KeyPgDn || k == tcell.KeyUp || k == tcell.KeyDown || k == tcell.KeyEnd || k == tcell.KeyHome { msgView := roomView.MessageView() if msgView.IsAtTop() && (k == tcell.KeyPgUp || k == tcell.KeyUp) { go view.LoadHistory(roomView.Room.ID) } switch k { case tcell.KeyPgUp: msgView.AddScrollOffset(msgView.Height() / 2) case tcell.KeyPgDn: msgView.AddScrollOffset(-msgView.Height() / 2) case tcell.KeyUp: msgView.AddScrollOffset(1) case tcell.KeyDown: msgView.AddScrollOffset(-1) case tcell.KeyHome: msgView.AddScrollOffset(msgView.TotalHeight()) case tcell.KeyEnd: msgView.AddScrollOffset(-msgView.TotalHeight()) } } else { return key } return nil } const WheelScrollOffsetDiff = 3 func isInArea(x, y int, p tview.Primitive) bool { rx, ry, rw, rh := p.GetRect() return x >= rx && y >= ry && x < rx+rw && y < ry+rh } func (view *MainView) MouseEventHandler(roomView *RoomView, event *tcell.EventMouse) *tcell.EventMouse { if event.Buttons() == tcell.ButtonNone || event.HasMotion() { return event } view.BumpFocus() msgView := roomView.MessageView() x, y := event.Position() switch { case isInArea(x, y, msgView): mx, my, _, _ := msgView.GetRect() switch event.Buttons() { case tcell.WheelUp: if msgView.IsAtTop() { go view.LoadHistory(roomView.Room.ID) } else { msgView.AddScrollOffset(WheelScrollOffsetDiff) view.parent.Render() } case tcell.WheelDown: msgView.AddScrollOffset(-WheelScrollOffsetDiff) view.parent.Render() if msgView.ScrollOffset == 0 { roomView.Room.MarkRead() } default: if msgView.HandleClick(x-mx, y-my, event.Buttons()) { view.parent.Render() } } case isInArea(x, y, view.roomList): switch event.Buttons() { case tcell.WheelUp: view.roomList.AddScrollOffset(-WheelScrollOffsetDiff) view.parent.Render() case tcell.WheelDown: view.roomList.AddScrollOffset(WheelScrollOffsetDiff) view.parent.Render() case tcell.Button1: _, rly, _, _ := msgView.GetRect() n := y - rly + 1 view.SwitchRoom(view.roomList.Get(n)) } default: debug.Print("Unhandled mouse event:", event.Buttons(), event.Modifiers(), x, y) } return event } func (view *MainView) SwitchRoom(tag string, room *rooms.Room) { if room == nil { return } view.roomView.SwitchToPage(room.ID) roomView := view.rooms[room.ID] if roomView.MessageView().ScrollOffset == 0 { roomView.Room.MarkRead() } view.roomList.SetSelected(tag, room) view.parent.app.SetFocus(view) view.parent.Render() } func (view *MainView) Focus(delegate func(p tview.Primitive)) { room := view.roomList.SelectedRoom() if room != nil { roomView, ok := view.rooms[room.ID] if ok { delegate(roomView) } } } 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) } } } func (view *MainView) addRoomPage(room *rooms.Room) { if !view.roomView.HasPage(room.ID) { roomView := NewRoomView(view, room). SetInputSubmitFunc(view.InputSubmit). SetInputChangedFunc(view.InputChanged). SetInputCapture(view.KeyEventHandler). SetMouseCapture(view.MouseEventHandler) view.rooms[room.ID] = roomView view.roomView.AddPage(room.ID, roomView, true, false) roomView.UpdateUserList() _, err := roomView.LoadHistory(view.matrix, view.config.HistoryDir) if err != nil { debug.Printf("Failed to load history of %s: %v", roomView.Room.GetTitle(), err) } } } func (view *MainView) GetRoom(roomID string) ifc.RoomView { room, ok := view.rooms[roomID] if !ok { view.AddRoom(roomID) room, ok := view.rooms[roomID] if !ok { return nil } return room } return room } func (view *MainView) AddRoom(roomID string) { if view.roomList.Contains(roomID) { debug.Print("Add aborted", roomID) return } debug.Print("Adding", roomID) room := view.matrix.GetRoom(roomID) view.roomList.Add(room) view.addRoomPage(room) if !view.roomList.HasSelected() { view.SwitchRoom(view.roomList.First()) } } func (view *MainView) RemoveRoom(roomID string) { roomView := view.GetRoom(roomID) if roomView == nil { debug.Print("Remove aborted", roomID) return } debug.Print("Removing", roomID) view.roomList.Remove(roomView.MxRoom()) view.SwitchRoom(view.roomList.Selected()) view.roomView.RemovePage(roomID) delete(view.rooms, roomID) view.parent.Render() } func (view *MainView) SetRooms(rooms map[string]*rooms.Room) { view.roomList.Clear() view.roomView.Clear() view.rooms = make(map[string]*RoomView) for _, room := range rooms { if room.HasLeft { continue } view.roomList.Add(room) view.addRoomPage(room) } view.SwitchRoom(view.roomList.First()) } func (view *MainView) UpdateTags(room *rooms.Room, newTags []rooms.RoomTag) { if len(newTags) == 0 { for _, tag := range room.RawTags { view.roomList.RemoveFromTag(tag.Tag, room) } view.roomList.AddToTag("", room) } else if len(room.RawTags) == 0 { view.roomList.RemoveFromTag("", room) for _, tag := range newTags { view.roomList.AddToTag(tag.Tag, room) } } else { NewTags: for _, newTag := range newTags { for _, oldTag := range room.RawTags { if newTag.Tag == oldTag.Tag { continue NewTags } } view.roomList.AddToTag(newTag.Tag, room) } OldTags: for _, oldTag := range room.RawTags { for _, newTag := range newTags { if newTag.Tag == oldTag.Tag { continue OldTags } } view.roomList.RemoveFromTag(oldTag.Tag, room) } } room.RawTags = newTags } func (view *MainView) SetTyping(room string, users []string) { roomView, ok := view.rooms[room] if ok { roomView.SetTyping(users) view.parent.Render() } } func sendNotification(room *rooms.Room, sender, text string, critical, sound bool) { if room.GetTitle() != sender { sender = fmt.Sprintf("%s (%s)", sender, room.GetTitle()) } debug.Printf("Sending notification with body \"%s\" from %s in room ID %s (critical=%v, sound=%v)", text, sender, room.ID, critical, sound) notification.Send(sender, text, critical, sound) } func (view *MainView) NotifyMessage(room *rooms.Room, message ifc.Message, should pushrules.PushActionArrayShould) { // Whether or not the room where the message came is the currently shown room. isCurrent := room == view.roomList.SelectedRoom() // Whether or not the terminal window is focused. isFocused := time.Now().Add(-30 * time.Second).Before(view.lastFocusTime) // Whether or not the push rules say this message should be notified about. shouldNotify := (should.Notify || !should.NotifySpecified) && message.Sender() != view.config.Session.UserID if !isCurrent { // The message is not in the current room, show new message status in room list. room.HasNewMessages = true room.Highlighted = should.Highlight || room.Highlighted if shouldNotify { room.UnreadMessages++ } } if shouldNotify && !isFocused { // Push rules say notify and the terminal is not focused, send desktop notification. shouldPlaySound := should.PlaySound && should.SoundName == "default" sendNotification(room, message.Sender(), message.NotificationContent(), should.Highlight, shouldPlaySound) } message.SetIsHighlight(should.Highlight) view.roomList.Bump(room) } func (view *MainView) LoadHistory(room string) { defer debug.Recover() roomView := view.rooms[room] batch := roomView.Room.PrevBatch lockTime := time.Now().Unix() + 1 roomView.Room.LockHistory() roomView.MessageView().LoadingMessages = true defer func() { roomView.Room.UnlockHistory() 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 } debug.Print("Fetching history for", room, "starting from", batch) history, prevBatch, err := view.matrix.GetHistory(roomView.Room.ID, batch, 50) if err != nil { roomView.AddServiceMessage("Failed to fetch history") debug.Print("Failed to fetch history for", roomView.Room.ID, err) return } roomView.Room.PrevBatch = prevBatch for _, evt := range history { message := view.ParseEvent(roomView, &evt) if message != nil { roomView.AddMessage(message, ifc.PrependMessage) } } err = roomView.SaveHistory(view.config.HistoryDir) if err != nil { debug.Printf("Failed to save history of %s: %v", roomView.Room.GetTitle(), err) } view.config.Session.Save() view.parent.Render() } func (view *MainView) ParseEvent(roomView ifc.RoomView, evt *gomatrix.Event) ifc.Message { return parser.ParseEvent(view.matrix, roomView.MxRoom(), evt) }