@ -25,6 +25,8 @@ var (
g * gocui . Gui
g * gocui . Gui
)
)
var config * Config
func main ( ) {
func main ( ) {
if ! k . LoggedIn {
if ! k . LoggedIn {
fmt . Println ( "You are not logged in." )
fmt . Println ( "You are not logged in." )
@ -37,7 +39,7 @@ func main() {
}
}
defer g . Close ( )
defer g . Close ( )
g . SetManagerFunc ( layout )
g . SetManagerFunc ( layout )
go RunCommand ( "config" , "load" )
RunCommand ( "config" , "load" )
go populateList ( )
go populateList ( )
go updateChatWindow ( )
go updateChatWindow ( )
if len ( os . Args ) > 1 {
if len ( os . Args ) > 1 {
@ -82,7 +84,7 @@ func layout(g *gocui.Gui) error {
chatView . Autoscroll = true
chatView . Autoscroll = true
chatView . Wrap = true
chatView . Wrap = true
welcomeText := basicStyle . stylize ( "Welcome $USER!\n\nYour chats will appear here.\nSupported commands are as follows:\n" )
welcomeText := basicStyle . stylize ( "Welcome $USER!\n\nYour chats will appear here.\nSupported commands are as follows:\n" )
welcomeText = welcomeText . replace ( "$USER" , mentionColor . stylize ( k . Username ) )
welcomeText = welcomeText . replace ( "$USER" , config . Colors . Message . Mention . stylize ( k . Username ) )
fmt . Fprintln ( chatView , welcomeText . string ( ) )
fmt . Fprintln ( chatView , welcomeText . string ( ) )
RunCommand ( "help" )
RunCommand ( "help" )
}
}
@ -95,7 +97,7 @@ func layout(g *gocui.Gui) error {
}
}
inputView . Editable = true
inputView . Editable = true
inputView . Wrap = true
inputView . Wrap = true
inputView . Title = fmt . Sprintf ( " Not in a chat - write `%sj` to join" , cmdPrefix )
inputView . Title = fmt . Sprintf ( " Not in a chat - write `%sj` to join" , config . Basics . C mdPrefix )
g . Cursor = true
g . Cursor = true
}
}
if listView , err4 := g . SetView ( "List" , 0 , 0 , maxX / 2 - maxX / 3 - 1 , maxY - 1 , 0 ) ; err4 != nil {
if listView , err4 := g . SetView ( "List" , 0 , 0 , maxX / 2 - maxX / 3 - 1 , maxY - 1 , 0 ) ; err4 != nil {
@ -255,7 +257,7 @@ func printError(message string) {
printErrorF ( message )
printErrorF ( message )
}
}
func printErrorF ( message string , parts ... StyledString ) {
func printErrorF ( message string , parts ... StyledString ) {
printToView ( "Feed" , errorCol or. sprintf ( removeFormatting ( message ) , parts ... ) . string ( ) )
printToView ( "Feed" , config . Colors . Feed . Err or. sprintf ( removeFormatting ( message ) , parts ... ) . string ( ) )
}
}
// this removes formatting
// this removes formatting
@ -265,7 +267,7 @@ func printInfo(message string) {
// this removes formatting
// this removes formatting
func printInfoF ( message string , parts ... StyledString ) {
func printInfoF ( message string , parts ... StyledString ) {
printToView ( "Feed" , feedColor . sprintf ( removeFormatting ( message ) , parts ... ) . string ( ) )
printToView ( "Feed" , config . Colors . Feed . Basic . sprintf ( removeFormatting ( message ) , parts ... ) . string ( ) )
}
}
func printToView ( viewName string , message string ) {
func printToView ( viewName string , message string ) {
g . Update ( func ( g * gocui . Gui ) error {
g . Update ( func ( g * gocui . Gui ) error {
@ -349,10 +351,10 @@ func populateList() {
log . Printf ( "%+v" , err )
log . Printf ( "%+v" , err )
} else {
} else {
clearView ( "List" )
clearView ( "List" )
var textBase = channelsColor . stylize ( "" )
var textBase = config . Colors . Channels . Basic . stylize ( "" )
var recentPMs = textBase . append ( channelsHeaderColo r . stylize ( "---[PMs]---\n" ) )
var recentPMs = textBase . append ( config . Colors . C hannels . Header . stylize ( "---[PMs]---\n" ) )
var recentPMsCount = 0
var recentPMsCount = 0
var recentChannels = textBase . append ( channelsHeaderColo r . stylize ( "---[Teams]---\n" ) )
var recentChannels = textBase . append ( config . Colors . C hannels . Header . stylize ( "---[Teams]---\n" ) )
var recentChannelsCount = 0
var recentChannelsCount = 0
for _ , s := range testVar . Result . Conversations {
for _ , s := range testVar . Result . Conversations {
channels = append ( channels , s . Channel )
channels = append ( channels , s . Channel )
@ -361,7 +363,7 @@ func populateList() {
if recentChannelsCount <= ( ( maxY - 2 ) / 3 ) {
if recentChannelsCount <= ( ( maxY - 2 ) / 3 ) {
channel := fmt . Sprintf ( "%s\n\t#%s\n" , s . Channel . Name , s . Channel . TopicName )
channel := fmt . Sprintf ( "%s\n\t#%s\n" , s . Channel . Name , s . Channel . TopicName )
if s . Unread {
if s . Unread {
recentChannels = recentChannels . append ( channelUnreadColor . stylize ( "*" + channel ) )
recentChannels = recentChannels . append ( config . Colors . C hannels . Unread . stylize ( "*" + channel ) )
} else {
} else {
recentChannels = recentChannels . appendString ( channel )
recentChannels = recentChannels . appendString ( channel )
}
}
@ -371,7 +373,7 @@ func populateList() {
if recentPMsCount <= ( ( maxY - 2 ) / 3 ) {
if recentPMsCount <= ( ( maxY - 2 ) / 3 ) {
pmName := fmt . Sprintf ( "%s\n" , cleanChannelName ( s . Channel . Name ) )
pmName := fmt . Sprintf ( "%s\n" , cleanChannelName ( s . Channel . Name ) )
if s . Unread {
if s . Unread {
recentPMs = recentPMs . append ( channelUnreadColor . stylize ( "*" + pmName ) )
recentPMs = recentPMs . append ( config . Colors . C hannels . Unread . stylize ( "*" + pmName ) )
} else {
} else {
recentPMs = recentPMs . appendString ( pmName )
recentPMs = recentPMs . appendString ( pmName )
}
}
@ -388,35 +390,35 @@ func populateList() {
// Formatting
// Formatting
func formatMessageBody ( body string ) StyledString {
func formatMessageBody ( body string ) StyledString {
output := messageBodyColor . stylize ( body )
output := config . Colors . Message . Body . stylize ( body )
output = colorReplaceMentionMe ( output )
output = colorReplaceMentionMe ( output )
output = output . colorRegex ( ` _[^_]*_ ` , messageBodyColor . withItalic ( ) )
output = output . colorRegex ( ` _[^_]*_ ` , config . Colors . Message . Body . withItalic ( ) )
output = output . colorRegex ( ` ~[^~]*~ ` , messageBodyColor . withStrikethrough ( ) )
output = output . colorRegex ( ` ~[^~]*~ ` , config . Colors . Message . Body . withStrikethrough ( ) )
output = output . colorRegex ( ` @[\w_]*(\.[\w_]+)* ` , messageLinkKeybaseColor )
output = output . colorRegex ( ` @[\w_]*(\.[\w_]+)* ` , config . Colors . Message . LinkKeybase )
// TODO change how bold, italic etc works, so it uses boldOn boldOff ([1m and [22m)
// TODO change how bold, italic etc works, so it uses boldOn boldOff ([1m and [22m)
output = output . colorRegex ( ` \*[^\*]*\* ` , messageBodyColor . withBold ( ) )
output = output . colorRegex ( ` \*[^\*]*\* ` , config . Colors . Message . Body . withBold ( ) )
output = output . replaceString ( "```" , "<code>" )
output = output . replaceString ( "```" , "<code>" )
// TODO make background color cover whole line
// TODO make background color cover whole line
output = output . colorRegex ( "<code>(.*\n)*<code>" , messageCodeColor )
output = output . colorRegex ( "<code>(.*\n)*<code>" , config . Colors . Message . Code )
output = output . colorRegex ( "`[^`]*`" , messageCodeColor )
output = output . colorRegex ( "`[^`]*`" , config . Colors . Message . Code )
// mention URL
// mention URL
output = output . colorRegex ( ` (https?:\/\/(www\.)?[-a-zA-Z0-9@:%._\+~#=] { 1,256}\.[a-zA-Z0-9()] { 1,6}\b([-a-zA-Z0-9()@:%_\+.~#?&//=]*)) ` , messageLinkURLColor )
output = output . colorRegex ( ` (https?:\/\/(www\.)?[-a-zA-Z0-9@:%._\+~#=] { 1,256}\.[a-zA-Z0-9()] { 1,6}\b([-a-zA-Z0-9()@:%_\+.~#?&//=]*)) ` , config . Colors . Message . LinkURL )
return output
return output
}
}
// TODO use this more
// TODO use this more
func formatChannel ( ch keybase . Channel ) StyledString {
func formatChannel ( ch keybase . Channel ) StyledString {
return messageLinkKeybaseColor . stylize ( fmt . Sprintf ( "@%s#%s" , ch . Name , ch . TopicName ) )
return config . Colors . Message . LinkKeybase . stylize ( fmt . Sprintf ( "@%s#%s" , ch . Name , ch . TopicName ) )
}
}
func colorReplaceMentionMe ( msg StyledString ) StyledString {
func colorReplaceMentionMe ( msg StyledString ) StyledString {
return msg . colorRegex ( ` (@?\b ` + k . Username + ` \b) ` , mentionColor )
return msg . colorRegex ( ` (@?\b ` + k . Username + ` \b) ` , config . Colors . Message . Mention )
}
}
func colorUsername ( username string ) StyledString {
func colorUsername ( username string ) StyledString {
var color = messageSenderDefaultColor
var color = config . Colors . Message . SenderDefault
if username == k . Username {
if username == k . Username {
color = mentionColor
color = config . Colors . Message . Mention
}
}
return color . stylize ( username )
return color . stylize ( username )
}
}
@ -427,27 +429,27 @@ func cleanChannelName(c string) string {
}
}
func formatMessage ( api keybase . ChatAPI , formatString string ) string {
func formatMessage ( api keybase . ChatAPI , formatString string ) string {
ret := messageHeaderColo r. stylize ( "" )
ret := config . Colors . Message . Heade r. stylize ( "" )
msgType := api . Msg . Content . Type
msgType := api . Msg . Content . Type
switch msgType {
switch msgType {
case "text" , "attachment" :
case "text" , "attachment" :
ret = messageHeaderColo r. stylize ( formatString )
ret = config . Colors . Message . Heade r. stylize ( formatString )
tm := time . Unix ( int64 ( api . Msg . SentAt ) , 0 )
tm := time . Unix ( int64 ( api . Msg . SentAt ) , 0 )
var msg = formatMessageBody ( api . Msg . Content . Text . Body )
var msg = formatMessageBody ( api . Msg . Content . Text . Body )
if msgType == "attachment" {
if msgType == "attachment" {
msg = messageBodyColor . stylize ( "$TITLE\n$FILE" )
msg = config . Colors . Message . Body . stylize ( "$TITLE\n$FILE" )
attachment := api . Msg . Content . Attachment
attachment := api . Msg . Content . Attachment
msg = msg . replaceString ( "$TITLE" , attachment . Object . Title )
msg = msg . replaceString ( "$TITLE" , attachment . Object . Title )
msg = msg . replace ( "$FILE" , messageAttachmentColor . stylize ( fmt . Sprintf ( "[Attachment: %s]" , attachment . Object . Filename ) ) )
msg = msg . replace ( "$FILE" , config . Colors . Message . Attachment . stylize ( fmt . Sprintf ( "[Attachment: %s]" , attachment . Object . Filename ) ) )
}
}
user := colorUsername ( api . Msg . Sender . Username )
user := colorUsername ( api . Msg . Sender . Username )
device := messageSenderDeviceColor . stylize ( api . Msg . Sender . DeviceName )
device := config . Colors . Message . SenderDevice . stylize ( api . Msg . Sender . DeviceName )
msgID := messageIDColor . stylize ( fmt . Sprintf ( "%d" , api . Msg . ID ) )
msgID := config . Colors . Message . ID . stylize ( fmt . Sprintf ( "%d" , api . Msg . ID ) )
date := messageTimeColor . stylize ( tm . Format ( d ateFormat) )
date := config . Colors . Message . Time . stylize ( tm . Format ( config . Formatting . D ateFormat) )
msgTime := messageTimeColor . stylize ( tm . Format ( timeFormat ) )
msgTime := config . Colors . Message . Time . stylize ( tm . Format ( config . Forma tting . T imeFormat) )
channelName := messageIDColor . stylize ( fmt . Sprintf ( "@%s#%s" , api . Msg . Channel . Name , api . Msg . Channel . TopicName ) )
channelName := config . Colors . Message . ID . stylize ( fmt . Sprintf ( "@%s#%s" , api . Msg . Channel . Name , api . Msg . Channel . TopicName ) )
ret = ret . replace ( "$MSG" , msg )
ret = ret . replace ( "$MSG" , msg )
ret = ret . replace ( "$USER" , user )
ret = ret . replace ( "$USER" , user )
ret = ret . replace ( "$DEVICE" , device )
ret = ret . replace ( "$DEVICE" , device )
@ -459,9 +461,9 @@ func formatMessage(api keybase.ChatAPI, formatString string) string {
return ret . string ( )
return ret . string ( )
}
}
func formatOutput ( api keybase . ChatAPI ) string {
func formatOutput ( api keybase . ChatAPI ) string {
format := outputFormat
format := c onfig . Formatting . O utputFormat
if stream {
if stream {
format = outputStreamFormat
format = c onfig . Formatting . O utputStreamFormat
}
}
return formatMessage ( api , format )
return formatMessage ( api , format )
}
}
@ -489,7 +491,7 @@ func handleMessage(api keybase.ChatAPI) {
if m . Text == k . Username {
if m . Text == k . Username {
// We are in a team
// We are in a team
if topicName != channel . TopicName {
if topicName != channel . TopicName {
printInfo ( formatMessage ( api , mentionFormat ) )
printInfo ( formatMessage ( api , config . For matting . OutputM entionFormat) )
fmt . Print ( "\a" )
fmt . Print ( "\a" )
}
}
@ -498,7 +500,7 @@ func handleMessage(api keybase.ChatAPI) {
}
}
} else {
} else {
if msgSender != channel . Name {
if msgSender != channel . Name {
printInfo ( formatMessage ( api , p mFormat) )
printInfo ( formatMessage ( api , config . For matting . PM Format) )
fmt . Print ( "\a" )
fmt . Print ( "\a" )
}
}
@ -516,7 +518,7 @@ func handleMessage(api keybase.ChatAPI) {
if api . Msg . Channel . MembersType == keybase . TEAM {
if api . Msg . Channel . MembersType == keybase . TEAM {
printToView ( "Chat" , formatOutput ( api ) )
printToView ( "Chat" , formatOutput ( api ) )
} else {
} else {
printToView ( "Chat" , formatMessage ( api , p mFormat) )
printToView ( "Chat" , formatMessage ( api , config . For matting . PM Format) )
}
}
}
}
} else {
} else {
@ -550,8 +552,8 @@ func handleInput(viewName string) error {
if inputString == "" {
if inputString == "" {
return nil
return nil
}
}
if strings . HasPrefix ( inputString , cmdPrefix ) {
if strings . HasPrefix ( inputString , config . Basics . C mdPrefix ) {
cmd := deleteEmpty ( strings . Split ( inputString [ len ( cmdPrefix ) : ] , " " ) )
cmd := deleteEmpty ( strings . Split ( inputString [ len ( config . Basics . C mdPrefix ) : ] , " " ) )
if len ( cmd ) < 1 {
if len ( cmd ) < 1 {
return nil
return nil
}
}