erroneousboat-slack-term/handlers/event.go

600 lines
15 KiB
Go
Raw Normal View History

2016-09-11 17:55:19 +02:00
package handlers
import (
"fmt"
2018-10-27 14:44:20 +02:00
"log"
"os"
"regexp"
2016-10-29 17:57:58 +02:00
"strconv"
"strings"
"time"
2018-04-01 13:03:28 +02:00
"github.com/0xAX/notificator"
2017-09-23 13:56:45 +02:00
"github.com/erroneousboat/termui"
"github.com/nlopes/slack"
termbox "github.com/nsf/termbox-go"
2016-09-11 17:55:19 +02:00
"github.com/erroneousboat/slack-term/components"
"github.com/erroneousboat/slack-term/config"
2016-10-19 09:08:31 +02:00
"github.com/erroneousboat/slack-term/context"
"github.com/erroneousboat/slack-term/views"
2016-09-11 17:55:19 +02:00
)
2018-04-01 13:03:28 +02:00
var scrollTimer *time.Timer
var notifyTimer *time.Timer
2016-10-29 17:57:58 +02:00
// actionMap binds specific action names to the function counterparts,
// these action names can then be used to bind them to specific keys
// in the Config.
var actionMap = map[string]func(*context.AppContext){
2017-12-17 13:48:20 +01:00
"space": actionSpace,
"backspace": actionBackSpace,
"delete": actionDelete,
"cursor-right": actionMoveCursorRight,
"cursor-left": actionMoveCursorLeft,
"send": actionSend,
"quit": actionQuit,
"mode-insert": actionInsertMode,
"mode-command": actionCommandMode,
"mode-search": actionSearchMode,
"clear-input": actionClearInput,
"channel-up": actionMoveCursorUpChannels,
"channel-down": actionMoveCursorDownChannels,
"channel-top": actionMoveCursorTopChannels,
"channel-bottom": actionMoveCursorBottomChannels,
"channel-search-next": actionSearchNextChannels,
"channel-search-prev": actionSearchPrevChannels,
"chat-up": actionScrollUpChat,
"chat-down": actionScrollDownChat,
"help": actionHelp,
}
2019-05-18 12:16:27 +02:00
// Initialize will start a combination of event handlers and 'background tasks'
func Initialize(ctx *context.AppContext) {
// Keyboard events
eventHandler(ctx)
// RTM incoming events
messageHandler(ctx)
// User presence
go actionSetPresenceAll(ctx)
2016-10-29 17:57:58 +02:00
}
2018-03-24 13:56:55 +01:00
// eventHandler will handle events created by the user
func eventHandler(ctx *context.AppContext) {
go func() {
for {
ctx.EventQueue <- termbox.PollEvent()
}
}()
go func() {
for {
ev := <-ctx.EventQueue
handleTermboxEvents(ctx, ev)
handleMoreTermboxEvents(ctx, ev)
2017-12-02 15:24:31 +01:00
// Place your debugging statements here
if ctx.Debug {
ctx.View.Debug.Println(
"event received",
)
2017-12-02 15:24:31 +01:00
}
}
}()
}
func handleTermboxEvents(ctx *context.AppContext, ev termbox.Event) bool {
switch ev.Type {
case termbox.EventKey:
actionKeyEvent(ctx, ev)
case termbox.EventResize:
actionResizeEvent(ctx, ev)
}
return true
}
func handleMoreTermboxEvents(ctx *context.AppContext, ev termbox.Event) bool {
for {
select {
2017-09-23 13:56:45 +02:00
case ev := <-ctx.EventQueue:
ok := handleTermboxEvents(ctx, ev)
if !ok {
return false
2016-09-11 17:55:19 +02:00
}
default:
return true
2016-09-11 17:55:19 +02:00
}
}
2016-09-11 17:55:19 +02:00
}
2018-03-24 13:56:55 +01:00
// messageHandler will handle events created by the service
func messageHandler(ctx *context.AppContext) {
2016-09-27 17:12:38 +02:00
go func() {
for {
select {
case msg := <-ctx.Service.RTM.IncomingEvents:
switch ev := msg.Data.(type) {
case *slack.MessageEvent:
2018-03-24 13:56:55 +01:00
2016-09-29 21:19:09 +02:00
// Construct message
2018-12-25 15:09:03 +01:00
msg, err := ctx.Service.CreateMessageFromMessageEvent(ev, ev.Channel)
2018-03-24 13:56:55 +01:00
if err != nil {
continue
}
2016-09-27 17:12:38 +02:00
// Add message to the selected channel
if ev.Channel == ctx.View.Channels.ChannelItems[ctx.View.Channels.SelectedChannel].ID {
2016-10-11 18:50:25 +02:00
2019-02-16 22:45:36 +01:00
// When timestamp isn't set this is a thread reply,
// handle as such
if ev.ThreadTimestamp != "" {
ctx.View.Chat.AddReply(ev.ThreadTimestamp, msg)
} else {
ctx.View.Chat.AddMessage(msg)
}
2016-09-27 22:05:44 +02:00
termui.Render(ctx.View.Chat)
2016-09-30 16:36:41 +02:00
2016-10-01 12:48:15 +02:00
// TODO: set Chat.Offset to 0, to automatically scroll
// down?
2016-09-27 17:12:38 +02:00
}
2016-09-30 12:09:03 +02:00
2016-10-01 12:48:15 +02:00
// Set new message indicator for channel, I'm leaving
// this here because I also want to be notified when
// I'm currently in a channel but not in the terminal
// window (tmux). But only create a notification when
// it comes from someone else but the current user.
2018-10-13 15:51:30 +02:00
if ev.User != ctx.Service.CurrentUserID {
actionNewMessage(ctx, ev)
}
case *slack.PresenceChangeEvent:
actionSetPresence(ctx, ev.User, ev.Presence)
case *slack.RTMError:
ctx.View.Debug.Println(
ev.Error(),
)
2016-09-27 17:12:38 +02:00
}
}
}
}()
}
func actionKeyEvent(ctx *context.AppContext, ev termbox.Event) {
keyStr := getKeyString(ev)
// Get the action name (actionStr) from the key that
// has been pressed. If this is found try to uncover
// the associated function with this key and execute
// it.
actionStr, ok := ctx.Config.KeyMap[ctx.Mode][keyStr]
if ok {
action, ok := actionMap[actionStr]
if ok {
action(ctx)
}
} else {
if ctx.Mode == context.InsertMode && ev.Ch != 0 {
actionInput(ctx.View, ev.Ch)
} else if ctx.Mode == context.SearchMode && ev.Ch != 0 {
actionSearch(ctx, ev.Ch)
}
}
}
func actionResizeEvent(ctx *context.AppContext, ev termbox.Event) {
2018-03-23 13:08:42 +01:00
// When terminal window is too small termui will panic, here
// we won't resize when the terminal window is too small.
if termui.TermWidth() < 25 || termui.TermHeight() < 5 {
return
}
2016-09-11 17:55:19 +02:00
termui.Body.Width = termui.TermWidth()
2018-01-27 12:03:10 +01:00
// Vertical resize components
ctx.View.Channels.List.Height = termui.TermHeight() - ctx.View.Input.Par.Height
ctx.View.Chat.List.Height = termui.TermHeight() - ctx.View.Input.Par.Height
ctx.View.Debug.List.Height = termui.TermHeight() - ctx.View.Input.Par.Height
2016-09-11 17:55:19 +02:00
termui.Body.Align()
termui.Render(termui.Body)
}
2016-10-21 21:23:25 +02:00
func actionInput(view *views.View, key rune) {
2016-09-11 17:55:19 +02:00
view.Input.Insert(key)
termui.Render(view.Input)
}
2017-07-15 23:56:39 +02:00
func actionClearInput(ctx *context.AppContext) {
// Clear input
ctx.View.Input.Clear()
ctx.View.Refresh()
// Set command mode
actionCommandMode(ctx)
}
2016-10-29 17:57:58 +02:00
func actionSpace(ctx *context.AppContext) {
actionInput(ctx.View, ' ')
}
func actionBackSpace(ctx *context.AppContext) {
ctx.View.Input.Backspace()
termui.Render(ctx.View.Input)
2016-10-09 14:59:48 +02:00
}
func actionDelete(ctx *context.AppContext) {
ctx.View.Input.Delete()
termui.Render(ctx.View.Input)
2016-09-11 17:55:19 +02:00
}
func actionMoveCursorRight(ctx *context.AppContext) {
ctx.View.Input.MoveCursorRight()
termui.Render(ctx.View.Input)
2016-09-11 17:55:19 +02:00
}
func actionMoveCursorLeft(ctx *context.AppContext) {
ctx.View.Input.MoveCursorLeft()
termui.Render(ctx.View.Input)
2016-09-11 17:55:19 +02:00
}
func actionSend(ctx *context.AppContext) {
if !ctx.View.Input.IsEmpty() {
// Clear message before sending, to combat
// quick succession of actionSend
2016-10-21 21:23:25 +02:00
message := ctx.View.Input.GetText()
ctx.View.Input.Clear()
ctx.View.Refresh()
2019-02-16 22:45:36 +01:00
// Send slash command
isCmd, err := ctx.Service.SendCommand(
ctx.View.Channels.ChannelItems[ctx.View.Channels.SelectedChannel].ID,
message,
2016-09-27 22:05:44 +02:00
)
if err != nil {
ctx.View.Debug.Println(
err.Error(),
)
}
2019-02-16 22:45:36 +01:00
// Send message
if !isCmd {
err := ctx.Service.SendMessage(
ctx.View.Channels.ChannelItems[ctx.View.Channels.SelectedChannel].ID,
message,
)
if err != nil {
ctx.View.Debug.Println(
err.Error(),
)
}
}
// Clear notification icon if there is any
channelItem := ctx.View.Channels.ChannelItems[ctx.View.Channels.SelectedChannel]
if channelItem.Notification {
2019-05-11 12:20:52 +02:00
ctx.Service.MarkAsRead(channelItem)
ctx.View.Channels.MarkAsRead(ctx.View.Channels.SelectedChannel)
}
termui.Render(ctx.View.Channels)
2016-09-11 17:55:19 +02:00
}
}
// actionSearch will search through the channels based on the users
// input. A time is implemented to make sure the actual searching
// and changing of channels is done when the user's typing is paused.
2017-07-15 23:56:39 +02:00
func actionSearch(ctx *context.AppContext, key rune) {
actionInput(ctx.View, key)
2017-07-15 23:56:39 +02:00
go func() {
2018-04-01 13:03:28 +02:00
if scrollTimer != nil {
scrollTimer.Stop()
2017-07-15 23:56:39 +02:00
}
2018-04-01 13:03:28 +02:00
scrollTimer = time.NewTimer(time.Second / 4)
<-scrollTimer.C
2017-07-15 23:56:39 +02:00
// Only actually search when the time expires
2017-07-15 23:56:39 +02:00
term := ctx.View.Input.GetText()
ctx.View.Channels.Search(term)
actionChangeChannel(ctx)
}()
}
// actionQuit will exit the program by using os.Exit, this is
// done because we are using a custom termui EvtStream. Which
// we won't be able to call termui.StopLoop() on. See main.go
// for the customEvtStream and why this is done.
func actionQuit(ctx *context.AppContext) {
termbox.Close()
os.Exit(0)
2016-09-11 17:55:19 +02:00
}
func actionInsertMode(ctx *context.AppContext) {
ctx.Mode = context.InsertMode
2017-12-02 11:09:01 +01:00
ctx.View.Mode.SetInsertMode()
2016-09-11 17:55:19 +02:00
}
func actionCommandMode(ctx *context.AppContext) {
ctx.Mode = context.CommandMode
2017-12-02 11:09:01 +01:00
ctx.View.Mode.SetCommandMode()
2016-09-11 17:55:19 +02:00
}
2016-09-25 22:34:02 +02:00
2017-07-15 23:56:39 +02:00
func actionSearchMode(ctx *context.AppContext) {
ctx.Mode = context.SearchMode
2017-12-02 11:09:01 +01:00
ctx.View.Mode.SetSearchMode()
2017-07-15 23:56:39 +02:00
}
2016-09-25 22:34:02 +02:00
func actionGetMessages(ctx *context.AppContext) {
2018-10-27 14:44:20 +02:00
msgs, err := ctx.Service.GetMessages(
ctx.View.Channels.ChannelItems[ctx.View.Channels.SelectedChannel].ID,
2017-12-01 23:52:25 +01:00
ctx.View.Chat.GetMaxItems(),
2016-09-27 22:05:44 +02:00
)
2018-10-27 14:44:20 +02:00
if err != nil {
termbox.Close()
log.Println(err)
os.Exit(0)
}
2017-12-03 21:43:33 +01:00
ctx.View.Chat.SetMessages(msgs)
2016-09-27 22:05:44 +02:00
2016-09-25 22:34:02 +02:00
termui.Render(ctx.View.Chat)
}
2017-12-01 23:52:25 +01:00
// actionMoveCursorUpChannels will execute the actionChangeChannel
// function. A timer is implemented to support fast scrolling through
2017-12-01 23:52:25 +01:00
// the list without executing the actionChangeChannel event
2016-09-27 22:05:44 +02:00
func actionMoveCursorUpChannels(ctx *context.AppContext) {
go func() {
2018-04-01 13:03:28 +02:00
if scrollTimer != nil {
scrollTimer.Stop()
}
ctx.View.Channels.MoveCursorUp()
termui.Render(ctx.View.Channels)
2018-04-01 13:03:28 +02:00
scrollTimer = time.NewTimer(time.Second / 4)
<-scrollTimer.C
2017-12-01 23:52:25 +01:00
// Only actually change channel when the timer expires
actionChangeChannel(ctx)
}()
2016-09-27 22:05:44 +02:00
}
2017-12-01 23:52:25 +01:00
// actionMoveCursorDownChannels will execute the actionChangeChannel
// function. A timer is implemented to support fast scrolling through
2017-12-01 23:52:25 +01:00
// the list without executing the actionChangeChannel event
2016-09-27 22:05:44 +02:00
func actionMoveCursorDownChannels(ctx *context.AppContext) {
go func() {
2018-04-01 13:03:28 +02:00
if scrollTimer != nil {
scrollTimer.Stop()
}
ctx.View.Channels.MoveCursorDown()
termui.Render(ctx.View.Channels)
2018-04-01 13:03:28 +02:00
scrollTimer = time.NewTimer(time.Second / 4)
<-scrollTimer.C
2017-12-01 23:52:25 +01:00
// Only actually change channel when the timer expires
actionChangeChannel(ctx)
}()
2016-09-29 19:09:30 +02:00
}
func actionMoveCursorTopChannels(ctx *context.AppContext) {
ctx.View.Channels.MoveCursorTop()
actionChangeChannel(ctx)
}
func actionMoveCursorBottomChannels(ctx *context.AppContext) {
ctx.View.Channels.MoveCursorBottom()
actionChangeChannel(ctx)
}
2017-12-17 13:48:20 +01:00
func actionSearchNextChannels(ctx *context.AppContext) {
ctx.View.Channels.SearchNext()
actionChangeChannel(ctx)
}
func actionSearchPrevChannels(ctx *context.AppContext) {
ctx.View.Channels.SearchPrev()
actionChangeChannel(ctx)
}
2016-09-29 19:09:30 +02:00
func actionChangeChannel(ctx *context.AppContext) {
// Clear messages from Chat pane
ctx.View.Chat.ClearMessages()
2016-09-27 22:05:44 +02:00
2017-12-01 23:52:25 +01:00
// Get messages of the SelectedChannel, and get the count of messages
// that fit into the Chat component
2018-10-27 14:44:20 +02:00
msgs, err := ctx.Service.GetMessages(
ctx.View.Channels.ChannelItems[ctx.View.Channels.SelectedChannel].ID,
2017-12-01 23:52:25 +01:00
ctx.View.Chat.GetMaxItems(),
2016-09-27 22:05:44 +02:00
)
2018-10-27 14:44:20 +02:00
if err != nil {
termbox.Close()
log.Println(err)
os.Exit(0)
}
2016-09-27 22:05:44 +02:00
2017-12-01 23:52:25 +01:00
// Set messages for the channel
ctx.View.Chat.SetMessages(msgs)
2017-12-01 23:52:25 +01:00
2016-09-29 19:09:30 +02:00
// Set channel name for the Chat pane
2016-09-28 22:10:04 +02:00
ctx.View.Chat.SetBorderLabel(
ctx.View.Channels.ChannelItems[ctx.View.Channels.SelectedChannel].GetChannelName(),
2016-09-28 22:10:04 +02:00
)
2017-12-01 23:52:25 +01:00
// Clear notification icon if there is any
channelItem := ctx.View.Channels.ChannelItems[ctx.View.Channels.SelectedChannel]
if channelItem.Notification {
2019-05-11 12:20:52 +02:00
ctx.Service.MarkAsRead(channelItem)
ctx.View.Channels.MarkAsRead(ctx.View.Channels.SelectedChannel)
}
2016-10-29 23:59:16 +02:00
2016-09-27 22:05:44 +02:00
termui.Render(ctx.View.Channels)
termui.Render(ctx.View.Chat)
}
2016-09-30 12:09:03 +02:00
// actionNewMessage will set the new message indicator for a channel, and
// if configured will also display a desktop notification
func actionNewMessage(ctx *context.AppContext, ev *slack.MessageEvent) {
ctx.View.Channels.MarkAsUnread(ev.Channel)
termui.Render(ctx.View.Channels)
2018-04-01 13:03:28 +02:00
// Terminal bell
fmt.Print("\a")
2018-04-01 13:03:28 +02:00
// Desktop notification
if ctx.Config.Notify == config.NotifyMention {
if isMention(ctx, ev) {
createNotifyMessage(ctx, ev)
}
} else if ctx.Config.Notify == config.NotifyAll {
createNotifyMessage(ctx, ev)
2018-04-01 13:03:28 +02:00
}
}
func actionSetPresence(ctx *context.AppContext, channelID string, presence string) {
ctx.View.Channels.SetPresence(channelID, presence)
2016-09-30 12:09:03 +02:00
termui.Render(ctx.View.Channels)
}
2016-09-30 16:36:41 +02:00
// actionPresenceAll will set the presence of the user list. Because the
// requests to the endpoint are rate limited we implement a timeout here.
func actionSetPresenceAll(ctx *context.AppContext) {
for _, chn := range ctx.Service.Conversations {
if chn.IsIM {
presence, err := ctx.Service.GetUserPresence(chn.User)
if err != nil {
presence = "away"
}
ctx.View.Channels.SetPresence(chn.ID, presence)
termui.Render(ctx.View.Channels)
time.Sleep(1200 * time.Millisecond)
}
}
}
2016-09-30 16:36:41 +02:00
func actionScrollUpChat(ctx *context.AppContext) {
ctx.View.Chat.ScrollUp()
termui.Render(ctx.View.Chat)
}
func actionScrollDownChat(ctx *context.AppContext) {
ctx.View.Chat.ScrollDown()
termui.Render(ctx.View.Chat)
}
2016-10-29 17:57:58 +02:00
2016-10-30 14:26:12 +01:00
func actionHelp(ctx *context.AppContext) {
2018-10-13 18:34:18 +02:00
ctx.View.Chat.Help(ctx.Usage, ctx.Config)
2016-10-30 14:26:12 +01:00
termui.Render(ctx.View.Chat)
}
2016-10-29 17:57:58 +02:00
// GetKeyString will return a string that resembles the key event from
// termbox. This is blatanly copied from termui because it is an unexported
// function.
//
// See:
// - https://github.com/gizak/termui/blob/a7e3aeef4cdf9fa2edb723b1541cb69b7bb089ea/events.go#L31-L72
// - https://github.com/nsf/termbox-go/blob/master/api_common.go
func getKeyString(e termbox.Event) string {
var ek string
k := string(e.Ch)
pre := ""
mod := ""
if e.Mod == termbox.ModAlt {
mod = "M-"
}
if e.Ch == 0 {
if e.Key > 0xFFFF-12 {
k = "<f" + strconv.Itoa(0xFFFF-int(e.Key)+1) + ">"
} else if e.Key > 0xFFFF-25 {
ks := []string{"<insert>", "<delete>", "<home>", "<end>", "<previous>", "<next>", "<up>", "<down>", "<left>", "<right>"}
k = ks[0xFFFF-int(e.Key)-12]
}
if e.Key <= 0x7F {
pre = "C-"
k = string('a' - 1 + int(e.Key))
kmap := map[termbox.Key][2]string{
termbox.KeyCtrlSpace: {"C-", "<space>"},
termbox.KeyBackspace: {"", "<backspace>"},
termbox.KeyTab: {"", "<tab>"},
termbox.KeyEnter: {"", "<enter>"},
termbox.KeyEsc: {"", "<escape>"},
termbox.KeyCtrlBackslash: {"C-", "\\"},
termbox.KeyCtrlSlash: {"C-", "/"},
termbox.KeySpace: {"", "<space>"},
termbox.KeyCtrl8: {"C-", "8"},
}
if sk, ok := kmap[e.Key]; ok {
pre = sk[0]
k = sk[1]
}
}
}
ek = pre + mod + k
return ek
}
// isMention check if the message event either contains a
// mention or is posted on an IM channel.
func isMention(ctx *context.AppContext, ev *slack.MessageEvent) bool {
channel := ctx.View.Channels.ChannelItems[ctx.View.Channels.FindChannel(ev.Channel)]
if channel.Type == components.ChannelTypeIM {
return true
}
// Mentions have the following format:
// <@U12345|erroneousboat>
// <@U12345>
r := regexp.MustCompile(`\<@(\w+\|*\w+)\>`)
matches := r.FindAllString(ev.Text, -1)
for _, match := range matches {
if strings.Contains(match, ctx.Service.CurrentUserID) {
return true
}
}
return false
}
func createNotifyMessage(ctx *context.AppContext, ev *slack.MessageEvent) {
go func() {
if notifyTimer != nil {
notifyTimer.Stop()
}
2018-09-02 10:14:34 +02:00
// Only actually notify when time expires
notifyTimer = time.NewTimer(time.Second * 2)
<-notifyTimer.C
var message string
channel := ctx.View.Channels.ChannelItems[ctx.View.Channels.FindChannel(ev.Channel)]
switch channel.Type {
case components.ChannelTypeChannel:
message = fmt.Sprintf("Message received on channel: %s", channel.Name)
case components.ChannelTypeGroup:
message = fmt.Sprintf("Message received in group: %s", channel.Name)
case components.ChannelTypeIM:
message = fmt.Sprintf("Message received from: %s", channel.Name)
default:
message = fmt.Sprintf("Message received from: %s", channel.Name)
}
ctx.Notify.Push("slack-term", message, "", notificator.UR_NORMAL)
}()
}