2018-03-17 01:27:30 +02: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/>.
|
|
|
|
|
2018-03-18 21:24:03 +02:00
|
|
|
package widget
|
2018-03-17 01:27:30 +02:00
|
|
|
|
|
|
|
import (
|
2018-03-22 19:51:20 +02:00
|
|
|
"encoding/gob"
|
2018-03-18 17:34:42 +02:00
|
|
|
"fmt"
|
2018-03-23 17:26:06 +02:00
|
|
|
"math"
|
2018-03-22 19:51:20 +02:00
|
|
|
"os"
|
2018-03-17 01:27:30 +02:00
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/gdamore/tcell"
|
2018-03-21 19:46:27 +02:00
|
|
|
"maunium.net/go/gomuks/ui/debug"
|
2018-03-18 21:24:03 +02:00
|
|
|
"maunium.net/go/gomuks/ui/types"
|
2018-03-17 01:27:30 +02:00
|
|
|
"maunium.net/go/tview"
|
|
|
|
)
|
|
|
|
|
|
|
|
type MessageView struct {
|
|
|
|
*tview.Box
|
|
|
|
|
|
|
|
ScrollOffset int
|
|
|
|
MaxSenderWidth int
|
2018-03-18 17:34:42 +02:00
|
|
|
DateFormat string
|
2018-03-17 01:27:30 +02:00
|
|
|
TimestampFormat string
|
|
|
|
TimestampWidth int
|
2018-03-20 12:16:32 +02:00
|
|
|
LoadingMessages bool
|
2018-03-17 01:27:30 +02:00
|
|
|
|
2018-03-20 12:16:32 +02:00
|
|
|
widestSender int
|
|
|
|
prevWidth int
|
|
|
|
prevHeight int
|
2018-03-20 17:26:28 +02:00
|
|
|
prevMsgCount int
|
2018-03-17 01:27:30 +02:00
|
|
|
|
2018-03-20 19:14:39 +02:00
|
|
|
messageIDs map[string]*types.Message
|
2018-03-18 21:24:03 +02:00
|
|
|
messages []*types.Message
|
2018-03-20 12:16:32 +02:00
|
|
|
|
|
|
|
textBuffer []string
|
2018-03-20 13:01:59 +02:00
|
|
|
metaBuffer []types.MessageMeta
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
|
|
|
|
2018-03-17 15:48:31 +02:00
|
|
|
func NewMessageView() *MessageView {
|
2018-03-17 01:27:30 +02:00
|
|
|
return &MessageView{
|
|
|
|
Box: tview.NewBox(),
|
2018-03-22 18:14:08 +02:00
|
|
|
MaxSenderWidth: 15,
|
2018-03-18 17:34:42 +02:00
|
|
|
DateFormat: "January _2, 2006",
|
2018-03-17 01:27:30 +02:00
|
|
|
TimestampFormat: "15:04:05",
|
|
|
|
TimestampWidth: 8,
|
|
|
|
ScrollOffset: 0,
|
|
|
|
|
2018-03-18 21:24:03 +02:00
|
|
|
messages: make([]*types.Message, 0),
|
2018-03-20 19:14:39 +02:00
|
|
|
messageIDs: make(map[string]*types.Message),
|
2018-03-20 12:16:32 +02:00
|
|
|
textBuffer: make([]string, 0),
|
2018-03-20 13:01:59 +02:00
|
|
|
metaBuffer: make([]types.MessageMeta, 0),
|
2018-03-18 17:34:42 +02:00
|
|
|
|
2018-03-20 12:16:32 +02:00
|
|
|
widestSender: 5,
|
|
|
|
prevWidth: -1,
|
|
|
|
prevHeight: -1,
|
2018-03-20 17:26:28 +02:00
|
|
|
prevMsgCount: -1,
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-22 21:44:46 +02:00
|
|
|
func (view *MessageView) NewMessage(id, sender, msgtype, text string, timestamp time.Time) *types.Message {
|
|
|
|
return types.NewMessage(id, sender, msgtype, text,
|
2018-03-18 17:34:42 +02:00
|
|
|
timestamp.Format(view.TimestampFormat),
|
|
|
|
timestamp.Format(view.DateFormat),
|
2018-03-18 21:24:03 +02:00
|
|
|
GetHashColor(sender))
|
2018-03-18 17:34:42 +02:00
|
|
|
}
|
|
|
|
|
2018-03-22 19:51:20 +02:00
|
|
|
func (view *MessageView) SaveHistory(path string) error {
|
|
|
|
file, err := os.OpenFile(path, os.O_WRONLY|os.O_CREATE, 0600)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
defer file.Close()
|
|
|
|
|
|
|
|
enc := gob.NewEncoder(file)
|
|
|
|
err = enc.Encode(view.messages)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *MessageView) LoadHistory(path string) (int, error) {
|
|
|
|
file, err := os.OpenFile(path, os.O_RDONLY, 0600)
|
|
|
|
if err != nil {
|
|
|
|
if os.IsNotExist(err) {
|
|
|
|
return 0, nil
|
|
|
|
}
|
|
|
|
return -1, err
|
|
|
|
}
|
|
|
|
defer file.Close()
|
|
|
|
|
|
|
|
dec := gob.NewDecoder(file)
|
|
|
|
err = dec.Decode(&view.messages)
|
|
|
|
if err != nil {
|
|
|
|
return -1, err
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, message := range view.messages {
|
|
|
|
view.updateWidestSender(message.Sender)
|
|
|
|
}
|
|
|
|
|
|
|
|
return len(view.messages), nil
|
|
|
|
}
|
|
|
|
|
2018-03-18 17:34:42 +02:00
|
|
|
func (view *MessageView) updateWidestSender(sender string) {
|
2018-03-17 01:27:30 +02:00
|
|
|
if len(sender) > view.widestSender {
|
|
|
|
view.widestSender = len(sender)
|
|
|
|
if view.widestSender > view.MaxSenderWidth {
|
|
|
|
view.widestSender = view.MaxSenderWidth
|
|
|
|
}
|
|
|
|
}
|
2018-03-18 17:34:42 +02:00
|
|
|
}
|
|
|
|
|
2018-03-20 19:14:39 +02:00
|
|
|
type MessageDirection int
|
|
|
|
|
2018-03-18 17:34:42 +02:00
|
|
|
const (
|
2018-04-01 10:28:46 +03:00
|
|
|
AppendMessage MessageDirection = iota
|
2018-03-18 17:34:42 +02:00
|
|
|
PrependMessage
|
2018-03-20 19:14:39 +02:00
|
|
|
IgnoreMessage
|
2018-03-18 17:34:42 +02:00
|
|
|
)
|
|
|
|
|
2018-03-20 19:14:39 +02:00
|
|
|
func (view *MessageView) UpdateMessageID(message *types.Message, newID string) {
|
|
|
|
delete(view.messageIDs, message.ID)
|
|
|
|
message.ID = newID
|
|
|
|
view.messageIDs[message.ID] = message
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *MessageView) AddMessage(message *types.Message, direction MessageDirection) {
|
2018-03-22 19:51:20 +02:00
|
|
|
if message == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-03-20 19:14:39 +02:00
|
|
|
msg, messageExists := view.messageIDs[message.ID]
|
2018-03-22 19:51:20 +02:00
|
|
|
if msg != nil && messageExists {
|
2018-03-20 19:14:39 +02:00
|
|
|
message.CopyTo(msg)
|
2018-03-22 21:44:46 +02:00
|
|
|
message = msg
|
2018-03-20 19:14:39 +02:00
|
|
|
direction = IgnoreMessage
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
2018-03-18 17:34:42 +02:00
|
|
|
|
|
|
|
view.updateWidestSender(message.Sender)
|
|
|
|
|
|
|
|
_, _, width, _ := view.GetInnerRect()
|
2018-03-17 01:27:30 +02:00
|
|
|
width -= view.TimestampWidth + TimestampSenderGap + view.widestSender + SenderMessageGap
|
2018-03-18 21:24:03 +02:00
|
|
|
message.CalculateBuffer(width)
|
2018-03-18 17:34:42 +02:00
|
|
|
|
|
|
|
if direction == AppendMessage {
|
|
|
|
if view.ScrollOffset > 0 {
|
2018-03-22 19:51:20 +02:00
|
|
|
view.ScrollOffset += message.Height()
|
2018-03-18 17:34:42 +02:00
|
|
|
}
|
|
|
|
view.messages = append(view.messages, message)
|
2018-03-20 17:26:28 +02:00
|
|
|
view.appendBuffer(message)
|
2018-03-18 17:34:42 +02:00
|
|
|
} else if direction == PrependMessage {
|
2018-03-18 21:24:03 +02:00
|
|
|
view.messages = append([]*types.Message{message}, view.messages...)
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
2018-03-18 17:34:42 +02:00
|
|
|
|
2018-03-20 19:14:39 +02:00
|
|
|
view.messageIDs[message.ID] = message
|
2018-03-20 12:16:32 +02:00
|
|
|
}
|
2018-03-17 01:27:30 +02:00
|
|
|
|
2018-03-20 12:16:32 +02:00
|
|
|
func (view *MessageView) appendBuffer(message *types.Message) {
|
|
|
|
if len(view.metaBuffer) > 0 {
|
|
|
|
prevMeta := view.metaBuffer[len(view.metaBuffer)-1]
|
2018-03-20 13:01:59 +02:00
|
|
|
if prevMeta != nil && prevMeta.GetDate() != message.Date {
|
2018-03-20 12:16:32 +02:00
|
|
|
view.textBuffer = append(view.textBuffer, fmt.Sprintf("Date changed to %s", message.Date))
|
2018-03-20 13:01:59 +02:00
|
|
|
view.metaBuffer = append(view.metaBuffer, &types.BasicMeta{TextColor: tcell.ColorGreen})
|
2018-03-20 12:16:32 +02:00
|
|
|
}
|
|
|
|
}
|
2018-03-17 01:27:30 +02:00
|
|
|
|
2018-03-22 19:51:20 +02:00
|
|
|
view.textBuffer = append(view.textBuffer, message.Buffer()...)
|
|
|
|
for range message.Buffer() {
|
2018-03-20 12:16:32 +02:00
|
|
|
view.metaBuffer = append(view.metaBuffer, message)
|
|
|
|
}
|
2018-03-20 17:26:28 +02:00
|
|
|
view.prevMsgCount++
|
2018-03-20 12:16:32 +02:00
|
|
|
}
|
|
|
|
|
2018-03-20 17:26:28 +02:00
|
|
|
func (view *MessageView) recalculateBuffers() {
|
2018-03-20 12:16:32 +02:00
|
|
|
_, _, width, height := view.GetInnerRect()
|
|
|
|
|
2018-03-20 17:26:28 +02:00
|
|
|
width -= view.TimestampWidth + TimestampSenderGap + view.widestSender + SenderMessageGap
|
|
|
|
recalculateMessageBuffers := width != view.prevWidth
|
|
|
|
if height != view.prevHeight || recalculateMessageBuffers || len(view.messages) != view.prevMsgCount {
|
|
|
|
view.textBuffer = []string{}
|
|
|
|
view.metaBuffer = []types.MessageMeta{}
|
|
|
|
view.prevMsgCount = 0
|
2018-03-20 12:16:32 +02:00
|
|
|
for _, message := range view.messages {
|
2018-03-20 17:26:28 +02:00
|
|
|
if recalculateMessageBuffers {
|
|
|
|
message.CalculateBuffer(width)
|
|
|
|
}
|
2018-03-20 12:16:32 +02:00
|
|
|
view.appendBuffer(message)
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
2018-03-20 12:16:32 +02:00
|
|
|
view.prevHeight = height
|
2018-03-20 17:26:28 +02:00
|
|
|
view.prevWidth = width
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-20 12:16:32 +02:00
|
|
|
const PaddingAtTop = 5
|
|
|
|
|
2018-03-25 12:35:50 +03:00
|
|
|
func (view *MessageView) AddScrollOffset(diff int) {
|
2018-03-17 01:27:30 +02:00
|
|
|
_, _, _, height := view.GetInnerRect()
|
2018-03-20 12:16:32 +02:00
|
|
|
|
|
|
|
totalHeight := len(view.textBuffer)
|
2018-03-26 14:31:44 +03:00
|
|
|
if diff >= 0 && view.ScrollOffset+diff >= totalHeight-height+PaddingAtTop {
|
|
|
|
view.ScrollOffset = totalHeight - height + PaddingAtTop
|
|
|
|
} else {
|
|
|
|
view.ScrollOffset += diff
|
2018-03-20 12:16:32 +02:00
|
|
|
}
|
|
|
|
|
2018-03-26 14:31:44 +03:00
|
|
|
if view.ScrollOffset > totalHeight-height+PaddingAtTop {
|
|
|
|
view.ScrollOffset = totalHeight - height + PaddingAtTop
|
|
|
|
}
|
|
|
|
if view.ScrollOffset < 0 {
|
2018-03-25 12:35:50 +03:00
|
|
|
view.ScrollOffset = 0
|
2018-03-20 12:16:32 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-25 12:35:50 +03:00
|
|
|
func (view *MessageView) Height() int {
|
2018-03-20 12:16:32 +02:00
|
|
|
_, _, _, height := view.GetInnerRect()
|
2018-03-25 12:35:50 +03:00
|
|
|
return height
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *MessageView) TotalHeight() int {
|
|
|
|
return len(view.textBuffer)
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
|
|
|
|
2018-03-25 12:35:50 +03:00
|
|
|
func (view *MessageView) IsAtTop() bool {
|
2018-03-17 01:27:30 +02:00
|
|
|
_, _, _, height := view.GetInnerRect()
|
2018-03-25 12:35:50 +03:00
|
|
|
totalHeight := len(view.textBuffer)
|
|
|
|
return view.ScrollOffset >= totalHeight-height+PaddingAtTop
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
TimestampSenderGap = 1
|
|
|
|
SenderSeparatorGap = 1
|
|
|
|
SenderMessageGap = 3
|
|
|
|
)
|
|
|
|
|
2018-03-26 14:31:44 +03:00
|
|
|
func getScrollbarStyle(scrollbarHere, isTop, isBottom bool) (char rune, style tcell.Style) {
|
|
|
|
char = '│'
|
|
|
|
style = tcell.StyleDefault
|
|
|
|
if scrollbarHere {
|
|
|
|
style = style.Foreground(tcell.ColorGreen)
|
|
|
|
}
|
|
|
|
if isTop {
|
|
|
|
if scrollbarHere {
|
|
|
|
char = '╥'
|
|
|
|
} else {
|
|
|
|
char = '┬'
|
|
|
|
}
|
|
|
|
} else if isBottom {
|
|
|
|
if scrollbarHere {
|
|
|
|
char = '╨'
|
|
|
|
} else {
|
|
|
|
char = '┴'
|
|
|
|
}
|
|
|
|
} else if scrollbarHere {
|
|
|
|
char = '║'
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-03-17 01:27:30 +02:00
|
|
|
func (view *MessageView) Draw(screen tcell.Screen) {
|
|
|
|
view.Box.Draw(screen)
|
|
|
|
|
2018-03-26 17:22:47 +03:00
|
|
|
x, y, _, height := view.GetInnerRect()
|
2018-03-20 17:26:28 +02:00
|
|
|
view.recalculateBuffers()
|
2018-03-18 17:34:42 +02:00
|
|
|
|
2018-03-20 12:16:32 +02:00
|
|
|
if len(view.textBuffer) == 0 {
|
2018-03-26 17:22:47 +03:00
|
|
|
writeLineSimple(screen, "It's quite empty in here.", x, y+height)
|
2018-03-18 17:34:42 +02:00
|
|
|
return
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
2018-03-18 17:34:42 +02:00
|
|
|
|
2018-03-23 17:26:06 +02:00
|
|
|
usernameX := x + view.TimestampWidth + TimestampSenderGap
|
|
|
|
messageX := usernameX + view.widestSender + SenderMessageGap
|
|
|
|
separatorX := usernameX + view.widestSender + SenderSeparatorGap
|
2018-03-17 01:27:30 +02:00
|
|
|
|
2018-03-20 12:16:32 +02:00
|
|
|
indexOffset := len(view.textBuffer) - view.ScrollOffset - height
|
|
|
|
if indexOffset <= -PaddingAtTop {
|
|
|
|
message := "Scroll up to load more messages."
|
|
|
|
if view.LoadingMessages {
|
|
|
|
message = "Loading more messages..."
|
2018-03-18 17:34:42 +02:00
|
|
|
}
|
2018-03-26 17:22:47 +03:00
|
|
|
writeLineSimpleColor(screen, message, messageX, y, tcell.ColorGreen)
|
2018-03-20 12:16:32 +02:00
|
|
|
}
|
2018-03-22 23:40:26 +02:00
|
|
|
|
2018-03-21 19:46:27 +02:00
|
|
|
if len(view.textBuffer) != len(view.metaBuffer) {
|
|
|
|
debug.ExtPrintf("Unexpected text/meta buffer length mismatch: %d != %d.", len(view.textBuffer), len(view.metaBuffer))
|
|
|
|
return
|
|
|
|
}
|
2018-03-22 23:40:26 +02:00
|
|
|
|
2018-03-26 14:31:44 +03:00
|
|
|
var scrollBarHeight, scrollBarPos int
|
|
|
|
// Black magic (aka math) used to figure out where the scroll bar should be put.
|
|
|
|
{
|
|
|
|
viewportHeight := float64(height)
|
|
|
|
contentHeight := float64(len(view.textBuffer))
|
|
|
|
|
|
|
|
scrollBarHeight = int(math.Ceil(viewportHeight / (contentHeight / viewportHeight)))
|
|
|
|
|
|
|
|
scrollBarPos = height - int(math.Round(float64(view.ScrollOffset)/contentHeight*viewportHeight))
|
|
|
|
}
|
2018-03-23 17:26:06 +02:00
|
|
|
|
2018-03-22 23:40:26 +02:00
|
|
|
var prevMeta types.MessageMeta
|
2018-03-23 17:26:06 +02:00
|
|
|
firstLine := true
|
2018-03-26 14:31:44 +03:00
|
|
|
skippedLines := 0
|
2018-03-23 17:26:06 +02:00
|
|
|
|
2018-03-20 12:16:32 +02:00
|
|
|
for line := 0; line < height; line++ {
|
|
|
|
index := indexOffset + line
|
|
|
|
if index < 0 {
|
2018-03-26 14:31:44 +03:00
|
|
|
skippedLines++
|
2018-03-20 12:16:32 +02:00
|
|
|
continue
|
2018-03-21 19:46:27 +02:00
|
|
|
} else if index >= len(view.textBuffer) {
|
2018-03-20 12:16:32 +02:00
|
|
|
break
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
2018-03-23 17:26:06 +02:00
|
|
|
|
2018-03-26 14:31:44 +03:00
|
|
|
showScrollbar := line-skippedLines >= scrollBarPos-scrollBarHeight && line-skippedLines < scrollBarPos
|
|
|
|
isTop := firstLine && view.ScrollOffset+height >= len(view.textBuffer)
|
|
|
|
isBottom := line == height-1 && view.ScrollOffset == 0
|
|
|
|
|
|
|
|
borderChar, borderStyle := getScrollbarStyle(showScrollbar, isTop, isBottom)
|
|
|
|
|
2018-03-23 17:26:06 +02:00
|
|
|
firstLine = false
|
2018-03-26 14:31:44 +03:00
|
|
|
|
2018-03-23 17:26:06 +02:00
|
|
|
screen.SetContent(separatorX, y+line, borderChar, nil, borderStyle)
|
|
|
|
|
2018-03-20 12:16:32 +02:00
|
|
|
text, meta := view.textBuffer[index], view.metaBuffer[index]
|
|
|
|
if meta != prevMeta {
|
2018-03-20 13:01:59 +02:00
|
|
|
if len(meta.GetTimestamp()) > 0 {
|
2018-03-26 17:22:47 +03:00
|
|
|
writeLineSimpleColor(screen, meta.GetTimestamp(), x, y+line, meta.GetTimestampColor())
|
2018-03-20 13:01:59 +02:00
|
|
|
}
|
2018-03-22 21:44:46 +02:00
|
|
|
if prevMeta == nil || meta.GetSender() != prevMeta.GetSender() {
|
2018-03-26 17:22:47 +03:00
|
|
|
writeLineColor(
|
2018-03-25 14:21:59 +03:00
|
|
|
screen, tview.AlignRight, meta.GetSender(),
|
|
|
|
usernameX, y+line, view.widestSender,
|
|
|
|
meta.GetSenderColor())
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
2018-03-20 12:16:32 +02:00
|
|
|
prevMeta = meta
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
2018-03-26 17:22:47 +03:00
|
|
|
writeLineSimpleColor(screen, text, messageX, y+line, meta.GetTextColor())
|
2018-03-17 01:27:30 +02:00
|
|
|
}
|
|
|
|
}
|