2018-03-15 19:45:52 +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-15 19:45:52 +02:00
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2018-03-22 19:51:20 +02:00
|
|
|
"path/filepath"
|
2018-03-22 21:44:46 +02:00
|
|
|
"strconv"
|
2018-03-15 19:45:52 +02:00
|
|
|
"strings"
|
2018-03-17 15:48:31 +02:00
|
|
|
"time"
|
2018-03-15 19:45:52 +02:00
|
|
|
|
|
|
|
"github.com/gdamore/tcell"
|
2018-03-22 23:46:43 +02:00
|
|
|
"maunium.net/go/gomuks/matrix/rooms"
|
2018-03-18 21:24:03 +02:00
|
|
|
"maunium.net/go/gomuks/ui/types"
|
2018-03-15 19:45:52 +02:00
|
|
|
"maunium.net/go/tview"
|
|
|
|
)
|
|
|
|
|
|
|
|
type RoomView struct {
|
|
|
|
*tview.Box
|
|
|
|
|
|
|
|
topic *tview.TextView
|
2018-03-17 01:27:30 +02:00
|
|
|
content *MessageView
|
2018-03-15 19:45:52 +02:00
|
|
|
status *tview.TextView
|
2018-03-15 20:33:01 +02:00
|
|
|
userList *tview.TextView
|
2018-03-22 17:36:06 +02:00
|
|
|
ulBorder *Border
|
2018-03-22 16:44:24 +02:00
|
|
|
input *AdvancedInputField
|
2018-03-18 21:24:03 +02:00
|
|
|
Room *rooms.Room
|
2018-03-15 20:33:01 +02:00
|
|
|
}
|
|
|
|
|
2018-03-20 12:16:32 +02:00
|
|
|
func NewRoomView(room *rooms.Room) *RoomView {
|
2018-03-15 19:45:52 +02:00
|
|
|
view := &RoomView{
|
2018-03-22 17:36:06 +02:00
|
|
|
Box: tview.NewBox(),
|
|
|
|
topic: tview.NewTextView(),
|
|
|
|
content: NewMessageView(),
|
|
|
|
status: tview.NewTextView(),
|
|
|
|
userList: tview.NewTextView(),
|
|
|
|
ulBorder: NewBorder(),
|
|
|
|
input: NewAdvancedInputField(),
|
|
|
|
Room: room,
|
2018-03-15 19:45:52 +02:00
|
|
|
}
|
2018-03-22 16:44:24 +02:00
|
|
|
|
|
|
|
view.input.
|
|
|
|
SetFieldBackgroundColor(tcell.ColorDefault).
|
|
|
|
SetPlaceholder("Send a message...").
|
|
|
|
SetPlaceholderExtColor(tcell.ColorGray)
|
|
|
|
|
2018-03-15 19:45:52 +02:00
|
|
|
view.topic.
|
2018-03-16 16:24:11 +02:00
|
|
|
SetText(strings.Replace(room.GetTopic(), "\n", " ", -1)).
|
2018-03-15 19:45:52 +02:00
|
|
|
SetBackgroundColor(tcell.ColorDarkGreen)
|
2018-03-22 16:44:24 +02:00
|
|
|
|
2018-03-15 19:45:52 +02:00
|
|
|
view.status.SetBackgroundColor(tcell.ColorDimGray)
|
2018-03-22 16:44:24 +02:00
|
|
|
|
2018-03-26 22:03:30 +03:00
|
|
|
view.userList.
|
|
|
|
SetDynamicColors(true).
|
|
|
|
SetWrap(false)
|
2018-03-22 16:44:24 +02:00
|
|
|
|
|
|
|
return view
|
|
|
|
}
|
|
|
|
|
2018-03-22 19:51:20 +02:00
|
|
|
func (view *RoomView) logPath(dir string) string {
|
|
|
|
return filepath.Join(dir, fmt.Sprintf("%s.gmxlog", view.Room.ID))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *RoomView) SaveHistory(dir string) error {
|
|
|
|
return view.MessageView().SaveHistory(view.logPath(dir))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *RoomView) LoadHistory(dir string) (int, error) {
|
|
|
|
return view.MessageView().LoadHistory(view.logPath(dir))
|
|
|
|
}
|
|
|
|
|
2018-03-22 16:44:24 +02:00
|
|
|
func (view *RoomView) SetTabCompleteFunc(fn func(room *RoomView, text string, cursorOffset int) string) *RoomView {
|
|
|
|
view.input.SetTabCompleteFunc(func(text string, cursorOffset int) string {
|
|
|
|
return fn(view, text, cursorOffset)
|
|
|
|
})
|
|
|
|
return view
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *RoomView) SetInputCapture(fn func(room *RoomView, event *tcell.EventKey) *tcell.EventKey) *RoomView {
|
|
|
|
view.input.SetInputCapture(func(event *tcell.EventKey) *tcell.EventKey {
|
|
|
|
return fn(view, event)
|
|
|
|
})
|
|
|
|
return view
|
|
|
|
}
|
|
|
|
|
2018-03-24 22:14:17 +02:00
|
|
|
func (view *RoomView) SetMouseCapture(fn func(room *RoomView, event *tcell.EventMouse) *tcell.EventMouse) *RoomView {
|
|
|
|
view.input.SetMouseCapture(func(event *tcell.EventMouse) *tcell.EventMouse {
|
|
|
|
return fn(view, event)
|
|
|
|
})
|
|
|
|
return view
|
|
|
|
}
|
|
|
|
|
2018-03-22 16:44:24 +02:00
|
|
|
func (view *RoomView) SetInputSubmitFunc(fn func(room *RoomView, text string)) *RoomView {
|
|
|
|
view.input.SetDoneFunc(func(key tcell.Key) {
|
|
|
|
if key == tcell.KeyEnter {
|
|
|
|
fn(view, view.input.GetText())
|
|
|
|
}
|
|
|
|
})
|
|
|
|
return view
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *RoomView) SetInputChangedFunc(fn func(room *RoomView, text string)) *RoomView {
|
|
|
|
view.input.SetChangedFunc(func(text string) {
|
|
|
|
fn(view, text)
|
|
|
|
})
|
|
|
|
return view
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *RoomView) SetInputText(newText string) *RoomView {
|
|
|
|
view.input.SetText(newText)
|
2018-03-15 19:45:52 +02:00
|
|
|
return view
|
|
|
|
}
|
|
|
|
|
2018-03-22 16:44:24 +02:00
|
|
|
func (view *RoomView) GetInputText() string {
|
|
|
|
return view.input.GetText()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *RoomView) GetInputField() *AdvancedInputField {
|
|
|
|
return view.input
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *RoomView) Focus(delegate func(p tview.Primitive)) {
|
|
|
|
delegate(view.input)
|
|
|
|
}
|
|
|
|
|
2018-03-22 18:14:08 +02:00
|
|
|
// Constants defining the size of the room view grid.
|
|
|
|
const (
|
|
|
|
UserListBorderWidth = 1
|
|
|
|
UserListWidth = 20
|
|
|
|
StaticHorizontalSpace = UserListBorderWidth + UserListWidth
|
|
|
|
|
|
|
|
TopicBarHeight = 1
|
|
|
|
StatusBarHeight = 1
|
|
|
|
InputBarHeight = 1
|
|
|
|
StaticVerticalSpace = TopicBarHeight + StatusBarHeight + InputBarHeight
|
|
|
|
)
|
|
|
|
|
2018-03-15 19:45:52 +02:00
|
|
|
func (view *RoomView) Draw(screen tcell.Screen) {
|
2018-03-22 18:14:08 +02:00
|
|
|
x, y, width, height := view.GetInnerRect()
|
|
|
|
if width <= 0 || height <= 0 {
|
|
|
|
return
|
|
|
|
}
|
2018-03-17 01:27:30 +02:00
|
|
|
|
2018-03-22 18:14:08 +02:00
|
|
|
// Calculate actual grid based on view rectangle and constants defined above.
|
|
|
|
var (
|
|
|
|
contentHeight = height - StaticVerticalSpace
|
|
|
|
contentWidth = width - StaticHorizontalSpace
|
|
|
|
|
|
|
|
userListBorderColumn = x + contentWidth
|
|
|
|
userListColumn = userListBorderColumn + UserListBorderWidth
|
|
|
|
|
|
|
|
topicRow = y
|
|
|
|
contentRow = topicRow + TopicBarHeight
|
|
|
|
statusRow = contentRow + contentHeight
|
|
|
|
inputRow = statusRow + StatusBarHeight
|
|
|
|
)
|
|
|
|
|
|
|
|
// Update the rectangles of all the children.
|
|
|
|
view.topic.SetRect(x, topicRow, width, TopicBarHeight)
|
|
|
|
view.content.SetRect(x, contentRow, contentWidth, contentHeight)
|
|
|
|
view.status.SetRect(x, statusRow, width, StatusBarHeight)
|
|
|
|
if userListColumn > x {
|
|
|
|
view.userList.SetRect(userListColumn, contentRow, UserListWidth, contentHeight)
|
|
|
|
view.ulBorder.SetRect(userListBorderColumn, contentRow, UserListBorderWidth, contentHeight)
|
|
|
|
}
|
|
|
|
view.input.SetRect(x, inputRow, width, InputBarHeight)
|
2018-03-15 19:45:52 +02:00
|
|
|
|
2018-03-22 18:14:08 +02:00
|
|
|
// Draw everything
|
|
|
|
view.Box.Draw(screen)
|
2018-03-15 19:45:52 +02:00
|
|
|
view.topic.Draw(screen)
|
|
|
|
view.content.Draw(screen)
|
|
|
|
view.status.Draw(screen)
|
2018-03-22 16:44:24 +02:00
|
|
|
view.input.Draw(screen)
|
2018-03-22 17:36:06 +02:00
|
|
|
view.ulBorder.Draw(screen)
|
2018-03-15 20:33:01 +02:00
|
|
|
view.userList.Draw(screen)
|
2018-03-15 19:45:52 +02:00
|
|
|
}
|
|
|
|
|
2018-03-18 21:24:03 +02:00
|
|
|
func (view *RoomView) SetStatus(status string) {
|
|
|
|
view.status.SetText(status)
|
|
|
|
}
|
|
|
|
|
2018-03-15 19:45:52 +02:00
|
|
|
func (view *RoomView) SetTyping(users []string) {
|
2018-03-16 16:24:11 +02:00
|
|
|
for index, user := range users {
|
2018-03-18 21:24:03 +02:00
|
|
|
member := view.Room.GetMember(user)
|
2018-03-16 16:24:11 +02:00
|
|
|
if member != nil {
|
|
|
|
users[index] = member.DisplayName
|
|
|
|
}
|
|
|
|
}
|
2018-03-15 19:45:52 +02:00
|
|
|
if len(users) == 0 {
|
|
|
|
view.status.SetText("")
|
|
|
|
} else if len(users) < 2 {
|
|
|
|
view.status.SetText("Typing: " + strings.Join(users, " and "))
|
|
|
|
} else {
|
|
|
|
view.status.SetText(fmt.Sprintf(
|
|
|
|
"Typing: %s and %s",
|
|
|
|
strings.Join(users[:len(users)-1], ", "), users[len(users)-1]))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-17 15:48:31 +02:00
|
|
|
func (view *RoomView) AutocompleteUser(existingText string) (completions []string) {
|
2018-03-20 16:03:05 +02:00
|
|
|
textWithoutPrefix := existingText
|
|
|
|
if strings.HasPrefix(existingText, "@") {
|
|
|
|
textWithoutPrefix = existingText[1:]
|
|
|
|
}
|
2018-03-18 21:24:03 +02:00
|
|
|
for _, user := range view.Room.GetMembers() {
|
2018-03-20 16:03:05 +02:00
|
|
|
if strings.HasPrefix(user.DisplayName, textWithoutPrefix) {
|
2018-03-17 15:48:31 +02:00
|
|
|
completions = append(completions, user.DisplayName)
|
|
|
|
} else if strings.HasPrefix(user.UserID, existingText) {
|
|
|
|
completions = append(completions, user.UserID)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2018-03-17 01:27:30 +02:00
|
|
|
func (view *RoomView) MessageView() *MessageView {
|
|
|
|
return view.content
|
|
|
|
}
|
2018-03-15 20:33:01 +02:00
|
|
|
|
2018-03-16 16:24:11 +02:00
|
|
|
func (view *RoomView) UpdateUserList() {
|
2018-03-17 15:48:31 +02:00
|
|
|
var joined strings.Builder
|
|
|
|
var invited strings.Builder
|
2018-03-18 21:24:03 +02:00
|
|
|
for _, user := range view.Room.GetMembers() {
|
2018-03-16 16:24:11 +02:00
|
|
|
if user.Membership == "join" {
|
2018-03-18 21:24:03 +02:00
|
|
|
joined.WriteString(AddHashColor(user.DisplayName))
|
2018-03-17 15:48:31 +02:00
|
|
|
joined.WriteRune('\n')
|
|
|
|
} else if user.Membership == "invite" {
|
2018-03-18 21:24:03 +02:00
|
|
|
invited.WriteString(AddHashColor(user.DisplayName))
|
2018-03-17 15:48:31 +02:00
|
|
|
invited.WriteRune('\n')
|
2018-03-16 16:24:11 +02:00
|
|
|
}
|
2018-03-15 21:38:43 +02:00
|
|
|
}
|
2018-03-17 15:48:31 +02:00
|
|
|
view.userList.Clear()
|
|
|
|
fmt.Fprintf(view.userList, "%s\n", joined.String())
|
|
|
|
if invited.Len() > 0 {
|
|
|
|
fmt.Fprintf(view.userList, "\nInvited:\n%s", invited.String())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-22 21:44:46 +02:00
|
|
|
func (view *RoomView) NewMessage(id, sender, msgtype, text string, timestamp time.Time) *types.Message {
|
2018-03-18 21:24:03 +02:00
|
|
|
member := view.Room.GetMember(sender)
|
2018-03-17 15:48:31 +02:00
|
|
|
if member != nil {
|
|
|
|
sender = member.DisplayName
|
|
|
|
}
|
2018-03-22 21:44:46 +02:00
|
|
|
return view.content.NewMessage(id, sender, msgtype, text, timestamp)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (view *RoomView) NewTempMessage(msgtype, text string) *types.Message {
|
|
|
|
now := time.Now()
|
|
|
|
id := strconv.FormatInt(now.UnixNano(), 10)
|
2018-03-23 17:45:58 +02:00
|
|
|
sender := ""
|
|
|
|
if ownerMember := view.Room.GetSessionOwner(); ownerMember != nil {
|
|
|
|
sender = ownerMember.DisplayName
|
|
|
|
}
|
2018-03-22 21:44:46 +02:00
|
|
|
message := view.NewMessage(id, sender, msgtype, text, now)
|
2018-03-22 23:47:59 +02:00
|
|
|
message.State = types.MessageStateSending
|
2018-03-22 21:44:46 +02:00
|
|
|
view.AddMessage(message, AppendMessage)
|
|
|
|
return message
|
2018-03-18 17:34:42 +02:00
|
|
|
}
|
|
|
|
|
2018-03-20 19:14:39 +02:00
|
|
|
func (view *RoomView) AddMessage(message *types.Message, direction MessageDirection) {
|
2018-03-18 17:34:42 +02:00
|
|
|
view.content.AddMessage(message, direction)
|
2018-03-15 19:45:52 +02:00
|
|
|
}
|