gomuks/ui/room-list.go

520 lines
12 KiB
Go
Raw Normal View History

2018-03-25 14:21:59 +03:00
// 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 <http://www.gnu.org/licenses/>.
package ui
2018-03-25 14:21:59 +03:00
import (
"fmt"
2018-04-24 02:13:17 +03:00
"regexp"
"strconv"
2018-04-24 02:13:17 +03:00
"strings"
"time"
"maunium.net/go/gomuks/debug"
2018-03-25 14:21:59 +03:00
"maunium.net/go/gomuks/matrix/rooms"
"maunium.net/go/gomuks/ui/widget"
2018-04-19 11:10:34 +03:00
"maunium.net/go/tcell"
2018-03-25 14:21:59 +03:00
"maunium.net/go/tview"
)
2018-04-24 02:13:17 +03:00
type roomListItem struct {
room *rooms.Room
priority float64
}
2018-03-25 14:21:59 +03:00
type RoomList struct {
*tview.Box
2018-04-24 02:13:17 +03:00
// The list of tags in display order.
tags []string
// The list of rooms, in reverse order.
2018-04-24 02:13:17 +03:00
items map[string][]*rooms.Room
// The selected room.
2018-04-24 02:13:17 +03:00
selected *rooms.Room
selectedTag string
2018-03-25 14:21:59 +03:00
scrollOffset int
2018-03-25 14:21:59 +03:00
// The item main text color.
mainTextColor tcell.Color
// The text color for selected items.
selectedTextColor tcell.Color
// The background color for selected items.
selectedBackgroundColor tcell.Color
}
func NewRoomList() *RoomList {
return &RoomList{
Box: tview.NewBox(),
2018-04-24 02:13:17 +03:00
items: make(map[string][]*rooms.Room),
2018-04-25 07:01:00 +03:00
tags: []string{"m.favourite", "net.maunium.gomuks.fake.direct", "", "m.lowpriority"},
2018-03-25 14:21:59 +03:00
scrollOffset: 0,
2018-03-25 14:21:59 +03:00
mainTextColor: tcell.ColorWhite,
selectedTextColor: tcell.ColorWhite,
selectedBackgroundColor: tcell.ColorDarkGreen,
}
}
func (list *RoomList) Contains(roomID string) bool {
2018-04-24 02:13:17 +03:00
for _, roomList := range list.items {
for _, room := range roomList {
if room.ID == roomID {
return true
}
}
}
return false
}
2018-03-25 14:21:59 +03:00
func (list *RoomList) Add(room *rooms.Room) {
2018-04-24 02:13:17 +03:00
for _, tag := range room.Tags() {
list.AddToTag(tag.Tag, room)
}
}
func (list *RoomList) CheckTag(tag string) {
index := list.IndexTag(tag)
items, ok := list.items[tag]
if ok && len(items) == 0 {
2018-04-24 02:13:17 +03:00
delete(list.items, tag)
ok = false
}
if ok && index == -1 {
list.tags = append(list.tags, tag)
} /* TODO this doesn't work properly
else if index != -1 {
2018-04-24 02:13:17 +03:00
list.tags = append(list.tags[0:index], list.tags[index+1:]...)
}*/
2018-04-24 02:13:17 +03:00
}
func (list *RoomList) AddToTag(tag string, room *rooms.Room) {
2018-04-25 07:01:00 +03:00
if tag == "" && len(room.GetMembers()) == 2 {
tag = "net.maunium.gomuks.fake.direct"
}
2018-04-24 02:13:17 +03:00
items, ok := list.items[tag]
if !ok {
list.items[tag] = []*rooms.Room{room}
return
}
// Add space for new item.
items = append(items, nil)
// The default insert index is the newly added slot.
// That index will be used if all other rooms in the list have the same LastReceivedMessage timestamp.
insertAt := len(items) - 1
// Find the spot where the new room should be put according to the last received message timestamps.
for i := 0; i < len(items)-1; i++ {
if items[i].LastReceivedMessage.After(room.LastReceivedMessage) {
insertAt = i
break
}
}
2018-04-24 02:13:17 +03:00
// Move newer rooms forward in the array.
for i := len(items) - 1; i > insertAt; i-- {
items[i] = items[i-1]
}
2018-04-24 02:13:17 +03:00
// Insert room.
items[insertAt] = room
list.items[tag] = items
list.CheckTag(tag)
2018-03-25 14:21:59 +03:00
}
func (list *RoomList) Remove(room *rooms.Room) {
2018-04-24 02:13:17 +03:00
for _, tag := range room.Tags() {
list.RemoveFromTag(tag.Tag, room)
}
}
func (list *RoomList) RemoveFromTag(tag string, room *rooms.Room) {
items, ok := list.items[tag]
if !ok {
return
}
index := list.indexInTag(tag, room)
if index == -1 {
return
}
items = append(items[0:index], items[index+1:]...)
if len(items) == 0 {
delete(list.items, tag)
} else {
list.items[tag] = items
}
if room == list.selected {
// Room is currently selected, move selection to another room.
if index > 0 {
list.selected = items[index-1]
} else if len(items) > 0 {
list.selected = items[0]
} else if len(list.items) > 0 {
for _, tag := range list.tags {
moreItems := list.items[tag]
if len(moreItems) > 0 {
list.selected = moreItems[0]
list.selectedTag = tag
}
}
2018-04-24 02:13:17 +03:00
} else {
list.selected = nil
list.selectedTag = ""
}
2018-03-25 14:21:59 +03:00
}
2018-04-24 02:13:17 +03:00
list.CheckTag(tag)
}
func (list *RoomList) Bump(room *rooms.Room) {
2018-04-24 02:13:17 +03:00
for _, tag := range room.Tags() {
list.bumpInTag(tag.Tag, room)
}
}
func (list *RoomList) bumpInTag(tag string, room *rooms.Room) {
items, ok := list.items[tag]
if !ok {
return
}
found := false
2018-04-24 02:13:17 +03:00
for i := 0; i < len(items)-1; i++ {
if items[i] == room {
found = true
}
if found {
2018-04-24 02:13:17 +03:00
items[i] = items[i+1]
}
2018-03-25 14:21:59 +03:00
}
2018-04-24 02:13:17 +03:00
if found {
items[len(items)-1] = room
room.LastReceivedMessage = time.Now()
}
2018-03-25 14:21:59 +03:00
}
func (list *RoomList) Clear() {
2018-04-24 02:13:17 +03:00
list.items = make(map[string][]*rooms.Room)
2018-03-25 14:21:59 +03:00
list.selected = nil
2018-04-24 02:13:17 +03:00
list.selectedTag = ""
2018-03-25 14:21:59 +03:00
}
2018-04-24 02:13:17 +03:00
func (list *RoomList) SetSelected(tag string, room *rooms.Room) {
2018-03-25 14:21:59 +03:00
list.selected = room
2018-04-24 16:51:40 +03:00
list.selectedTag = tag
pos := list.index(tag, room)
_, _, _, height := list.GetRect()
if pos <= list.scrollOffset {
list.scrollOffset = pos-1
} else if pos >= list.scrollOffset+height {
list.scrollOffset = pos-height+1
}
2018-04-25 07:01:00 +03:00
debug.Print("Selecting", room.GetTitle(), "in", list.GetTagDisplayName(tag))
2018-03-25 14:21:59 +03:00
}
func (list *RoomList) HasSelected() bool {
return list.selected != nil
}
2018-04-24 02:13:17 +03:00
func (list *RoomList) Selected() (string, *rooms.Room) {
return list.selectedTag, list.selected
}
func (list *RoomList) SelectedRoom() *rooms.Room {
return list.selected
}
func (list *RoomList) AddScrollOffset(offset int) {
list.scrollOffset += offset
if list.scrollOffset < 0 {
list.scrollOffset = 0
}
_, _, _, viewHeight := list.GetRect()
contentHeight := list.ContentHeight()
if list.scrollOffset > contentHeight-viewHeight {
list.scrollOffset = contentHeight - viewHeight
}
}
2018-04-24 02:13:17 +03:00
func (list *RoomList) First() (string, *rooms.Room) {
for _, tag := range list.tags {
items := list.items[tag]
if len(items) > 0 {
return tag, items[len(items)-1]
2018-04-24 02:13:17 +03:00
}
}
return "", nil
}
func (list *RoomList) Last() (string, *rooms.Room) {
for tagIndex := len(list.tags) - 1; tagIndex >= 0; tagIndex-- {
tag := list.tags[tagIndex]
items := list.items[tag]
if len(items) > 0 {
return tag, items[0]
2018-04-24 02:13:17 +03:00
}
}
return "", nil
}
func (list *RoomList) IndexTag(tag string) int {
for index, entry := range list.tags {
if tag == entry {
return index
}
}
return -1
}
func (list *RoomList) Previous() (string, *rooms.Room) {
if len(list.items) == 0 {
2018-04-24 02:13:17 +03:00
return "", nil
} else if list.selected == nil {
2018-04-24 02:13:17 +03:00
return list.First()
}
2018-04-24 02:13:17 +03:00
items := list.items[list.selectedTag]
index := list.indexInTag(list.selectedTag, list.selected)
2018-04-24 16:51:40 +03:00
if index == -1 {
return list.First()
} else if index == len(items)-1 {
2018-04-24 02:13:17 +03:00
tagIndex := list.IndexTag(list.selectedTag)
tagIndex--
for ; tagIndex >= 0; tagIndex-- {
prevTag := list.tags[tagIndex]
prevTagItems := list.items[prevTag]
if len(prevTagItems) > 0 {
return prevTag, prevTagItems[0]
2018-04-24 02:13:17 +03:00
}
}
return list.Last()
}
2018-04-24 02:13:17 +03:00
return list.selectedTag, items[index+1]
}
2018-04-24 02:13:17 +03:00
func (list *RoomList) Next() (string, *rooms.Room) {
if len(list.items) == 0 {
2018-04-24 02:13:17 +03:00
return "", nil
} else if list.selected == nil {
2018-04-24 02:13:17 +03:00
return list.First()
}
2018-04-24 02:13:17 +03:00
items := list.items[list.selectedTag]
index := list.indexInTag(list.selectedTag, list.selected)
2018-04-24 16:51:40 +03:00
if index == -1 {
return list.Last()
} else if index == 0 {
2018-04-24 02:13:17 +03:00
tagIndex := list.IndexTag(list.selectedTag)
tagIndex++
for ; tagIndex < len(list.tags); tagIndex++ {
nextTag := list.tags[tagIndex]
nextTagItems := list.items[nextTag]
if len(nextTagItems) > 0 {
return nextTag, nextTagItems[len(nextTagItems)-1]
2018-04-24 02:13:17 +03:00
}
}
return list.First()
}
2018-04-24 02:13:17 +03:00
return list.selectedTag, items[index-1]
}
2018-04-24 02:13:17 +03:00
func (list *RoomList) indexInTag(tag string, room *rooms.Room) int {
roomIndex := -1
2018-04-24 02:13:17 +03:00
items := list.items[tag]
for index, entry := range items {
if entry == room {
roomIndex = index
break
}
}
return roomIndex
}
func (list *RoomList) index(tag string, room *rooms.Room) int {
tagIndex := list.IndexTag(tag)
if tagIndex == -1 {
return -1
}
localIndex := list.indexInTag(tag, room)
if localIndex == -1 {
return -1
}
localIndex = len(list.items[tag]) - 1 - localIndex
// Tag header
localIndex += 1
if tagIndex > 0 {
for i := 0; i < tagIndex; i++ {
previousTag := list.tags[i]
previousItems := list.items[previousTag]
tagDisplayName := list.GetTagDisplayName(previousTag)
if len(tagDisplayName) > 0 {
// Previous tag header + space
localIndex += 2
// Previous tag items
localIndex += len(previousItems)
}
}
}
return localIndex
}
func (list *RoomList) ContentHeight() (height int) {
for _, tag := range list.tags {
items := list.items[tag]
tagDisplayName := list.GetTagDisplayName(tag)
if len(tagDisplayName) == 0 {
continue
}
height += 2 + len(items)
}
return
}
2018-04-24 02:13:17 +03:00
func (list *RoomList) Get(n int) (string, *rooms.Room) {
n += list.scrollOffset
2018-04-24 02:13:17 +03:00
if n < 0 {
return "", nil
}
for _, tag := range list.tags {
// Tag header
n--
items := list.items[tag]
if n < 0 {
return "", nil
} else if n < len(items) {
return tag, items[len(items)-1-n]
}
// Tag items
n -= len(items)
2018-04-24 16:51:40 +03:00
// Tag footer
n--
2018-04-24 02:13:17 +03:00
}
return "", nil
}
var nsRegex = regexp.MustCompile("^[a-z]\\.[a-z](?:\\.[a-z])*$")
func (list *RoomList) GetTagDisplayName(tag string) string {
switch {
case len(tag) == 0:
return "Rooms"
case tag == "m.favourite":
return "Favorites"
case tag == "m.lowpriority":
return "Low Priority"
2018-04-25 07:01:00 +03:00
case tag == "net.maunium.gomuks.fake.direct":
return "People"
2018-04-24 02:13:17 +03:00
case strings.HasPrefix(tag, "u."):
return tag[len("u."):]
case !nsRegex.MatchString(tag):
return tag
default:
return ""
}
}
2018-03-25 14:21:59 +03:00
// Draw draws this primitive onto the screen.
func (list *RoomList) Draw(screen tcell.Screen) {
list.Box.Draw(screen)
x, y, width, height := list.GetInnerRect()
bottomLimit := y + height
handledOffset := 0
2018-03-25 14:21:59 +03:00
// Draw the list items.
2018-04-24 02:13:17 +03:00
for _, tag := range list.tags {
items := list.items[tag]
2018-04-25 07:01:00 +03:00
tagDisplayName := list.GetTagDisplayName(tag)
if len(tagDisplayName) == 0 {
continue
}
localOffset := 0
if handledOffset < list.scrollOffset {
if handledOffset+len(items) < list.scrollOffset {
handledOffset += len(items) + 2
continue
} else {
localOffset = list.scrollOffset - handledOffset
handledOffset += localOffset
}
}
2018-04-25 07:01:00 +03:00
widget.WriteLine(screen, tview.AlignLeft, tagDisplayName, x, y, width, tcell.StyleDefault.Underline(true).Bold(true))
2018-04-24 02:13:17 +03:00
y++
for i := len(items) - 1; i >= 0; i-- {
item := items[i]
index := len(items) - 1 - i
2018-03-25 14:21:59 +03:00
2018-04-24 02:13:17 +03:00
if y >= bottomLimit {
break
}
2018-03-25 14:21:59 +03:00
if index < localOffset {
continue
}
2018-04-24 02:13:17 +03:00
text := item.GetTitle()
2018-03-25 14:21:59 +03:00
2018-04-24 02:13:17 +03:00
lineWidth := width
2018-04-24 02:13:17 +03:00
style := tcell.StyleDefault.Foreground(list.mainTextColor)
if tag == list.selectedTag && item == list.selected {
style = style.Foreground(list.selectedTextColor).Background(list.selectedBackgroundColor)
}
2018-04-24 02:13:17 +03:00
if item.HasNewMessages {
style = style.Bold(true)
2018-03-25 14:21:59 +03:00
}
2018-04-24 02:13:17 +03:00
if item.UnreadMessages > 0 {
unreadMessageCount := "99+"
if item.UnreadMessages < 100 {
unreadMessageCount = strconv.Itoa(item.UnreadMessages)
}
if item.Highlighted {
unreadMessageCount += "!"
}
unreadMessageCount = fmt.Sprintf("(%s)", unreadMessageCount)
widget.WriteLine(screen, tview.AlignRight, unreadMessageCount, x+lineWidth-6, y, 6, style)
lineWidth -= len(unreadMessageCount) + 1
}
2018-04-24 02:13:17 +03:00
widget.WriteLine(screen, tview.AlignLeft, text, x, y, lineWidth, style)
y++
if y >= bottomLimit {
break
}
2018-03-25 14:21:59 +03:00
}
2018-04-24 16:51:40 +03:00
y++
2018-03-25 14:21:59 +03:00
}
}