Move input field to RoomView

This commit is contained in:
Tulir Asokan 2018-03-22 16:44:24 +02:00
parent 9fd67102ad
commit 492a8752f1
3 changed files with 105 additions and 53 deletions

View File

@ -39,7 +39,6 @@ type GomuksUI interface {
} }
type MainView interface { type MainView interface {
InputTabComplete(text string, cursorOffset int) string
GetRoom(roomID string) *widget.RoomView GetRoom(roomID string) *widget.RoomView
HasRoom(roomID string) bool HasRoom(roomID string) bool
AddRoom(roomID string) AddRoom(roomID string)

View File

@ -41,7 +41,6 @@ type MainView struct {
roomList *tview.List roomList *tview.List
roomView *tview.Pages roomView *tview.Pages
rooms map[string]*widget.RoomView rooms map[string]*widget.RoomView
input *widget.AdvancedInputField
currentRoomIndex int currentRoomIndex int
roomIDs []string roomIDs []string
@ -52,7 +51,7 @@ type MainView struct {
} }
func (view *MainView) addItem(p tview.Primitive, x, y, w, h int) { func (view *MainView) addItem(p tview.Primitive, x, y, w, h int) {
view.Grid.AddItem(p, x, y, w, h, 0, 0, false) view.Grid.AddItem(p, y, x, w, h, 0, 0, false)
} }
func (ui *GomuksUI) NewMainView() tview.Primitive { func (ui *GomuksUI) NewMainView() tview.Primitive {
@ -61,7 +60,6 @@ func (ui *GomuksUI) NewMainView() tview.Primitive {
roomList: tview.NewList(), roomList: tview.NewList(),
roomView: tview.NewPages(), roomView: tview.NewPages(),
rooms: make(map[string]*widget.RoomView), rooms: make(map[string]*widget.RoomView),
input: widget.NewAdvancedInputField(),
matrix: ui.gmx.MatrixContainer(), matrix: ui.gmx.MatrixContainer(),
gmx: ui.gmx, gmx: ui.gmx,
@ -69,7 +67,7 @@ func (ui *GomuksUI) NewMainView() tview.Primitive {
parent: ui, parent: ui,
} }
mainView.SetColumns(30, 1, 0).SetRows(0, 1) mainView.SetColumns(30, 1, 0).SetRows(0)
mainView.roomList. mainView.roomList.
ShowSecondaryText(false). ShowSecondaryText(false).
@ -77,30 +75,20 @@ func (ui *GomuksUI) NewMainView() tview.Primitive {
SetSelectedTextColor(tcell.ColorWhite). SetSelectedTextColor(tcell.ColorWhite).
SetBorderPadding(0, 0, 1, 0) SetBorderPadding(0, 0, 1, 0)
mainView.input. mainView.addItem(mainView.roomList, 0, 0, 1, 1)
SetDoneFunc(mainView.InputDone). mainView.addItem(widget.NewBorder(), 1, 0, 1, 1)
SetChangedFunc(mainView.InputChanged). mainView.addItem(mainView.roomView, 2, 0, 1, 1)
SetTabCompleteFunc(mainView.InputTabComplete).
SetFieldBackgroundColor(tcell.ColorDefault).
SetPlaceholder("Send a message...").
SetPlaceholderExtColor(tcell.ColorGray).
SetInputCapture(mainView.InputCapture)
mainView.addItem(mainView.roomList, 0, 0, 2, 1)
mainView.addItem(widget.NewBorder(), 0, 1, 2, 1)
mainView.addItem(mainView.roomView, 0, 2, 1, 1)
mainView.AddItem(mainView.input, 1, 2, 1, 1, 0, 0, true)
ui.mainView = mainView ui.mainView = mainView
return mainView return mainView
} }
func (view *MainView) InputChanged(text string) { func (view *MainView) InputChanged(roomView *widget.RoomView, text string) {
if len(text) == 0 { if len(text) == 0 {
go view.matrix.SendTyping(view.CurrentRoomID(), false) go view.matrix.SendTyping(roomView.Room.ID, false)
} else if text[0] != '/' { } else if text[0] != '/' {
go view.matrix.SendTyping(view.CurrentRoomID(), true) go view.matrix.SendTyping(roomView.Room.ID, true)
} }
} }
@ -116,42 +104,35 @@ func findWordToTabComplete(text string) string {
return output return output
} }
func (view *MainView) InputTabComplete(text string, cursorOffset int) string { func (view *MainView) InputTabComplete(roomView *widget.RoomView, text string, cursorOffset int) string {
roomView, _ := view.rooms[view.CurrentRoomID()] str := runewidth.Truncate(text, cursorOffset, "")
if roomView != nil { word := findWordToTabComplete(str)
str := runewidth.Truncate(text, cursorOffset, "") userCompletions := roomView.AutocompleteUser(word)
word := findWordToTabComplete(str) if len(userCompletions) == 1 {
userCompletions := roomView.AutocompleteUser(word) startIndex := len(str) - len(word)
if len(userCompletions) == 1 { completion := userCompletions[0]
startIndex := len(str) - len(word) if startIndex == 0 {
completion := userCompletions[0] completion = completion + ": "
if startIndex == 0 {
completion = completion + ": "
}
text = str[0:startIndex] + completion + text[len(str):]
} else if len(userCompletions) > 1 && len(userCompletions) < 6 {
roomView.SetStatus(fmt.Sprintf("Completions: %s", strings.Join(userCompletions, ", ")))
} }
text = str[0:startIndex] + completion + text[len(str):]
} else if len(userCompletions) > 1 && len(userCompletions) < 6 {
roomView.SetStatus(fmt.Sprintf("Completions: %s", strings.Join(userCompletions, ", ")))
} }
return text return text
} }
func (view *MainView) InputDone(key tcell.Key) { func (view *MainView) InputSubmit(roomView *widget.RoomView, text string) {
if key != tcell.KeyEnter {
return
}
room, text := view.CurrentRoomID(), view.input.GetText()
if len(text) == 0 { if len(text) == 0 {
return return
} else if text[0] == '/' { } else if text[0] == '/' {
args := strings.SplitN(text, " ", 2) args := strings.SplitN(text, " ", 2)
command := strings.ToLower(args[0]) command := strings.ToLower(args[0])
args = args[1:] args = args[1:]
go view.HandleCommand(room, command, args) go view.HandleCommand(roomView.Room.ID, command, args)
} else { } else {
view.SendMessage(room, text) view.SendMessage(roomView.Room.ID, text)
} }
view.input.SetText("") roomView.SetInputText("")
} }
func (view *MainView) SendMessage(room, text string) { func (view *MainView) SendMessage(room, text string) {
@ -205,7 +186,7 @@ func (view *MainView) HandleCommand(room, command string, args []string) {
} }
} }
func (view *MainView) InputCapture(key *tcell.EventKey) *tcell.EventKey { func (view *MainView) InputKeyHandler(roomView *widget.RoomView, key *tcell.EventKey) *tcell.EventKey {
k := key.Key() k := key.Key()
if key.Modifiers() == tcell.ModCtrl || key.Modifiers() == tcell.ModAlt { if key.Modifiers() == tcell.ModCtrl || key.Modifiers() == tcell.ModAlt {
if k == tcell.KeyDown { if k == tcell.KeyDown {
@ -218,10 +199,10 @@ func (view *MainView) InputCapture(key *tcell.EventKey) *tcell.EventKey {
return key return key
} }
} else if k == tcell.KeyPgUp || k == tcell.KeyPgDn || k == tcell.KeyUp || k == tcell.KeyDown { } else if k == tcell.KeyPgUp || k == tcell.KeyPgDn || k == tcell.KeyUp || k == tcell.KeyDown {
msgView := view.rooms[view.CurrentRoomID()].MessageView() msgView := roomView.MessageView()
if k == tcell.KeyPgUp || k == tcell.KeyUp { if k == tcell.KeyPgUp || k == tcell.KeyUp {
if msgView.IsAtTop() { if msgView.IsAtTop() {
go view.LoadMoreHistory(view.CurrentRoomID()) go view.LoadMoreHistory(roomView.Room.ID)
} else { } else {
msgView.MoveUp(k == tcell.KeyPgUp) msgView.MoveUp(k == tcell.KeyPgUp)
} }
@ -252,9 +233,17 @@ func (view *MainView) SwitchRoom(roomIndex int) {
view.currentRoomIndex = roomIndex % len(view.roomIDs) view.currentRoomIndex = roomIndex % len(view.roomIDs)
view.roomView.SwitchToPage(view.CurrentRoomID()) view.roomView.SwitchToPage(view.CurrentRoomID())
view.roomList.SetCurrentItem(roomIndex) view.roomList.SetCurrentItem(roomIndex)
view.gmx.App().SetFocus(view)
view.parent.Render() view.parent.Render()
} }
func (view *MainView) Focus(delegate func(p tview.Primitive)) {
roomView, ok := view.rooms[view.CurrentRoomID()]
if ok {
delegate(roomView)
}
}
func (view *MainView) addRoom(index int, room string) { func (view *MainView) addRoom(index int, room string) {
roomStore := view.matrix.GetRoom(room) roomStore := view.matrix.GetRoom(room)
@ -262,7 +251,11 @@ func (view *MainView) addRoom(index int, room string) {
view.SwitchRoom(index) view.SwitchRoom(index)
}) })
if !view.roomView.HasPage(room) { if !view.roomView.HasPage(room) {
roomView := widget.NewRoomView(roomStore) roomView := widget.NewRoomView(roomStore).
SetInputSubmitFunc(view.InputSubmit).
SetInputChangedFunc(view.InputChanged).
SetTabCompleteFunc(view.InputTabComplete).
SetInputCapture(view.InputKeyHandler)
view.rooms[room] = roomView view.rooms[room] = roomView
view.roomView.AddPage(room, roomView, true, false) view.roomView.AddPage(room, roomView, true, false)
roomView.UpdateUserList() roomView.UpdateUserList()

View File

@ -23,7 +23,7 @@ import (
"time" "time"
"github.com/gdamore/tcell" "github.com/gdamore/tcell"
rooms "maunium.net/go/gomuks/matrix/room" "maunium.net/go/gomuks/matrix/room"
"maunium.net/go/gomuks/ui/types" "maunium.net/go/gomuks/ui/types"
"maunium.net/go/tview" "maunium.net/go/tview"
) )
@ -35,6 +35,7 @@ type RoomView struct {
content *MessageView content *MessageView
status *tview.TextView status *tview.TextView
userList *tview.TextView userList *tview.TextView
input *AdvancedInputField
Room *rooms.Room Room *rooms.Room
FetchHistoryLock *sync.Mutex FetchHistoryLock *sync.Mutex
@ -47,33 +48,92 @@ func NewRoomView(room *rooms.Room) *RoomView {
content: NewMessageView(), content: NewMessageView(),
status: tview.NewTextView(), status: tview.NewTextView(),
userList: tview.NewTextView(), userList: tview.NewTextView(),
input: NewAdvancedInputField(),
FetchHistoryLock: &sync.Mutex{}, FetchHistoryLock: &sync.Mutex{},
Room: room, Room: room,
} }
view.input.
SetFieldBackgroundColor(tcell.ColorDefault).
SetPlaceholder("Send a message...").
SetPlaceholderExtColor(tcell.ColorGray)
view.topic. view.topic.
SetText(strings.Replace(room.GetTopic(), "\n", " ", -1)). SetText(strings.Replace(room.GetTopic(), "\n", " ", -1)).
SetBackgroundColor(tcell.ColorDarkGreen) SetBackgroundColor(tcell.ColorDarkGreen)
view.status.SetBackgroundColor(tcell.ColorDimGray) view.status.SetBackgroundColor(tcell.ColorDimGray)
view.userList.SetDynamicColors(true) view.userList.SetDynamicColors(true)
return view return view
} }
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
}
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)
return view
}
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)
}
func (view *RoomView) Draw(screen tcell.Screen) { func (view *RoomView) Draw(screen tcell.Screen) {
view.Box.Draw(screen) view.Box.Draw(screen)
x, y, width, height := view.GetRect() x, y, width, height := view.GetRect()
view.topic.SetRect(x, y, width, 1) view.topic.SetRect(x, y, width, 1)
view.content.SetRect(x, y+1, width-30, height-2) view.content.SetRect(x, y+1, width-30, height-3)
view.status.SetRect(x, y+height-1, width, 1) view.status.SetRect(x, y+height-2, width, 1)
view.userList.SetRect(x+width-29, y+1, 29, height-2) view.userList.SetRect(x+width-29, y+1, 29, height-3)
view.input.SetRect(x, y+height-1, width, 1)
view.topic.Draw(screen) view.topic.Draw(screen)
view.content.Draw(screen) view.content.Draw(screen)
view.status.Draw(screen) view.status.Draw(screen)
view.input.Draw(screen)
borderX := x + width - 30 borderX := x + width - 30
background := tcell.StyleDefault.Background(view.GetBackgroundColor()).Foreground(view.GetBorderColor()) background := tcell.StyleDefault.Background(view.GetBackgroundColor()).Foreground(view.GetBorderColor())
for borderY := y + 1; borderY < y+height-1; borderY++ { for borderY := y + 1; borderY < y+height-2; borderY++ {
screen.SetContent(borderX, borderY, tview.GraphicsVertBar, nil, background) screen.SetContent(borderX, borderY, tview.GraphicsVertBar, nil, background)
} }
view.userList.Draw(screen) view.userList.Draw(screen)