1
0
Fork 0
forked from Mirrors/gomuks
nyxmuks/ui/view-roster.go

338 lines
7.9 KiB
Go

// gomuks - A terminal Matrix client written in Go.
// Copyright (C) 2020 Tulir Asokan
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU Affero 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 Affero General Public License for more details.
//
// 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/>.
package ui
import (
"time"
sync "github.com/sasha-s/go-deadlock"
"go.mau.fi/mauview"
"go.mau.fi/tcell"
"maunium.net/go/gomuks/config"
"maunium.net/go/gomuks/matrix/rooms"
"maunium.net/go/gomuks/ui/widget"
"maunium.net/go/mautrix/event"
)
type RosterView struct {
mauview.Component
sync.RWMutex
selected *rooms.Room
rooms []*rooms.Room
scrollOffset int
height, width int
focused bool
parent *MainView
}
func NewRosterView(mainView *MainView) *RosterView {
rstr := &RosterView{
parent: mainView,
rooms: make([]*rooms.Room, 0),
}
return rstr
}
func (rstr *RosterView) Add(room *rooms.Room) {
if room.IsReplaced() {
return
}
rstr.Lock()
defer rstr.Unlock()
insertAt := len(rstr.rooms)
for i := 0; i < len(rstr.rooms); i++ {
if rstr.rooms[i] == room {
return
} else if room.LastReceivedMessage.After(rstr.rooms[i].LastReceivedMessage) {
insertAt = i
break
}
}
rstr.rooms = append(rstr.rooms, nil)
copy(rstr.rooms[insertAt+1:], rstr.rooms[insertAt:len(rstr.rooms)-1])
rstr.rooms[insertAt] = room
}
func (rstr *RosterView) Remove(room *rooms.Room) {
index := rstr.index(room)
if index < 0 || index > len(rstr.rooms) {
return
}
rstr.Lock()
defer rstr.Unlock()
last := len(rstr.rooms) - 1
if index < last {
copy(rstr.rooms[index:], rstr.rooms[index+1:])
}
rstr.rooms[last] = nil
rstr.rooms = rstr.rooms[:last]
}
func (rstr *RosterView) Bump(room *rooms.Room) {
rstr.Remove(room)
rstr.Add(room)
}
func (rstr *RosterView) index(room *rooms.Room) int {
rstr.Lock()
defer rstr.Unlock()
for index, entry := range rstr.rooms {
if entry == room {
return index
}
}
return -1
}
func (rstr *RosterView) getMostRecentMessage(room *rooms.Room) (string, bool) {
roomView, _ := rstr.parent.getRoomView(room.ID, true)
if msgView := roomView.MessageView(); len(msgView.messages) < 20 && !msgView.initialHistoryLoaded {
msgView.initialHistoryLoaded = true
go rstr.parent.LoadHistory(room.ID)
}
if len(roomView.content.messages) > 0 {
for index := len(roomView.content.messages) - 1; index >= 0; index-- {
if roomView.content.messages[index].Type == event.MsgText {
return roomView.content.messages[index].PlainText(), true
}
}
}
return "It's quite empty in here.", false
}
func (rstr *RosterView) First() *rooms.Room {
rstr.Lock()
defer rstr.Unlock()
return rstr.rooms[0]
}
func (rstr *RosterView) Last() *rooms.Room {
rstr.Lock()
defer rstr.Unlock()
return rstr.rooms[len(rstr.rooms)-1]
}
func (rstr *RosterView) ScrollNext() {
if index := rstr.index(rstr.selected); index == -1 || index == len(rstr.rooms)-1 {
rstr.selected = rstr.First()
rstr.scrollOffset = 0
} else {
rstr.Lock()
defer rstr.Unlock()
rstr.selected = rstr.rooms[index+1]
if rstr.VisualScrollHeight(rstr.scrollOffset, index+1) >= rstr.height {
rstr.scrollOffset++
}
}
}
func (rstr *RosterView) ScrollPrev() {
if index := rstr.index(rstr.selected); index < 1 {
rstr.selected = rstr.Last()
if i := len(rstr.rooms) - rstr.RoomsOnScreen(); i < 0 {
rstr.scrollOffset = 0
} else {
rstr.scrollOffset = i
}
} else {
rstr.Lock()
defer rstr.Unlock()
rstr.selected = rstr.rooms[index-1]
if index == rstr.scrollOffset {
rstr.scrollOffset--
}
}
}
func (rstr *RosterView) VisualScrollHeight(start, end int) int {
if start < 0 || start > end {
return -1
}
return 3 + (2 * (end - start))
}
func (rstr *RosterView) RoomsOnScreen() int {
return (rstr.height - 3) / 2
}
func (rstr *RosterView) IndexOfLastVisibleRoom() int {
return rstr.scrollOffset + rstr.RoomsOnScreen()
}
func (rstr *RosterView) Draw(screen mauview.Screen) {
if rstr.focused {
if roomView, ok := rstr.parent.getRoomView(rstr.selected.ID, true); ok {
roomView.Update()
roomView.Draw(screen)
return
}
}
rstr.width, rstr.height = screen.Size()
titleStyle := tcell.StyleDefault.Foreground(tcell.ColorDefault).Bold(true)
mainStyle := titleStyle.Bold(false)
now := time.Now()
tm := now.Format("15:04")
tmX := rstr.width - 3 - len(tm)
// first line
widget.WriteLine(screen, mauview.AlignLeft, "GOMUKS", 2, 1, tmX, titleStyle)
widget.WriteLine(screen, mauview.AlignLeft, tm, tmX, 1, 2+len(tm), titleStyle)
// second line
widget.WriteLine(screen, mauview.AlignRight, now.Format("Mon, Jan 02"), 0, 2, rstr.width-3, mainStyle)
// third line
widget.NewBorder().Draw(mauview.NewProxyScreen(screen, 2, 3, rstr.width-5, 1))
y := 4
for _, room := range rstr.rooms[rstr.scrollOffset:] {
if room.IsReplaced() {
continue
}
renderHeight := 2
if y+renderHeight >= rstr.height {
renderHeight = rstr.height - y
}
isSelected := room == rstr.selected
style := tcell.StyleDefault.
Foreground(tcell.ColorDefault).
Bold(room.HasNewMessages())
if isSelected {
style = style.
Foreground(tcell.ColorBlack).
Background(tcell.ColorWhite)
}
timestamp := room.LastReceivedMessage
tm := timestamp.Format("15:04")
now := time.Now()
today := time.Date(now.Year(), now.Month(), now.Day(), 0, 0, 0, 0, now.Location())
if timestamp.Before(today) {
if timestamp.Before(today.AddDate(0, 0, -6)) {
tm = timestamp.Format("2006-01-02")
} else {
tm = timestamp.Format("Monday")
}
}
lastMessage, received := rstr.getMostRecentMessage(room)
msgStyle := style.Foreground(tcell.ColorGray).Italic(!received)
if isSelected {
msgStyle = msgStyle.Background(tcell.ColorWhite)
}
tmX := rstr.width - 3 - len(tm)
widget.WriteLinePadded(screen, mauview.AlignLeft, room.GetTitle(), 2, y, tmX, style)
widget.WriteLine(screen, mauview.AlignLeft, tm, tmX, y, 2+len(tm), style)
widget.WriteLinePadded(screen, mauview.AlignLeft, lastMessage, 2, y+1, rstr.width-5, msgStyle)
y += renderHeight
if y >= rstr.height {
break
}
}
}
func (rstr *RosterView) OnKeyEvent(event mauview.KeyEvent) bool {
kb := config.Keybind{
Key: event.Key(),
Ch: event.Rune(),
Mod: event.Modifiers(),
}
if rstr.focused {
if rstr.parent.config.Keybindings.Roster[kb] == "clear" {
rstr.focused = false
rstr.selected = nil
} else {
if roomView, ok := rstr.parent.getRoomView(rstr.selected.ID, true); ok {
return roomView.OnKeyEvent(event)
}
}
}
switch rstr.parent.config.Keybindings.Roster[kb] {
case "next_room":
rstr.ScrollNext()
case "prev_room":
rstr.ScrollPrev()
case "clear":
rstr.selected = nil
case "quit":
rstr.parent.gmx.Stop(true)
case "enter":
rstr.focused = rstr.selected != nil
default:
return false
}
return true
}
func (rstr *RosterView) OnMouseEvent(event mauview.MouseEvent) bool {
if rstr.focused {
if roomView, ok := rstr.parent.getRoomView(rstr.selected.ID, true); ok {
return roomView.OnMouseEvent(event)
}
}
if event.HasMotion() {
return false
}
switch event.Buttons() {
case tcell.WheelUp:
rstr.ScrollPrev()
return true
case tcell.WheelDown:
rstr.ScrollNext()
return true
case tcell.Button1:
_, y := event.Position()
if y <= 3 || y > rstr.VisualScrollHeight(rstr.scrollOffset, rstr.IndexOfLastVisibleRoom()) {
return false
} else {
index := rstr.scrollOffset + y/2 - 2
if index > len(rstr.rooms)-1 {
return false
}
rstr.selected = rstr.rooms[index]
rstr.focused = true
return true
}
}
return false
}