1
0
mirror of https://github.com/fazo96/telecommander.git synced 2025-01-25 13:54:19 +01:00

improved interface, implemented marking messages as read

This commit is contained in:
Enrico Fasoli 2015-09-01 15:30:51 +02:00
parent d6ff917688
commit 66d9667c88
4 changed files with 163 additions and 188 deletions

View File

@ -14,14 +14,13 @@ As of __now__, __Telecommander lets you__:
- Chat in groups and in private (it also marks messages as read when you read them) - Chat in groups and in private (it also marks messages as read when you read them)
- Know when you have unread messages - Know when you have unread messages
- Do everything __completely in the terminal, even via ssh__ using only the keyboard - Do everything __completely in the terminal, even via ssh__ using only the keyboard
- Do everything (except writing, duh) using only the mouse! - Do everything (except writing messages, duh) using only the mouse, in the terminal!
- Navigate your contact and chat list - Navigate your contact and chat list
- Scroll back to view old messages - Scroll back to view old messages
Waht's missing for version _0.1.0_ (__Almost done!__): Waht's missing for version _0.1.0_ (__Almost done!__):
- View who's online and when was the last time they logged in - View who's online and when was the last time they logged in
- Testing and tying up loose ends in user interface and code quality
What's missing (for future versions up to 1.0) What's missing (for future versions up to 1.0)
@ -35,7 +34,6 @@ What's missing (for future versions up to 1.0)
- Sign out (without having to manually delete files) - Sign out (without having to manually delete files)
- Delete account - Delete account
- Reply to and Forward message support - Reply to and Forward message support
- Cool interface (it's already ok, just not as cool as I want it to be)
- Search, Tab completion, chatbox history - Search, Tab completion, chatbox history
- Send multiline messages - Send multiline messages
- Themes and configurability! Basic scripting! - Themes and configurability! Basic scripting!

161
lib/ui.js
View File

@ -8,31 +8,37 @@ module.exports = function(data){
// Get msgBox for given group/user NAME, create if not exists // Get msgBox for given group/user NAME, create if not exists
data.getMsgBox = function(chat){ data.getMsgBox = function(chat){
if(chat === undefined){ if(chat === undefined){
log('ERROR: asked for box for "undefined"!!') data.log('ERROR: asked for box for "undefined"!!')
return data.msgBox[statusWindow] return data.msgBox[statusWindow]
} }
if(!data.msgBox[chat]){ if(!data.msgBox[chat]){
//log('Generating window: "'+chat+'"') //log('Generating window: "'+chat+'"')
data.msgBox[chat] = data.mkBox(chat) data.msgBox[chat] = data.mkBox(chat)
data.screen.append(data.msgBox[chat]) data.screen.insertBefore(data.msgBox[chat],data.loader)
//data.getMessages(chat,data.msgBox[chat]) //data.getMessages(chat,data.msgBox[chat])
} // else log('Getting window','"'+chat+'"') } // else log('Getting window','"'+chat+'"')
return data.msgBox[chat] return data.msgBox[chat]
} }
data.switchToBox = function(boxname){ data.switchToBox = function(boxname){
// Hide current window
if(data.selectedWindow && data.msgBox[data.selectedWindow]) if(data.selectedWindow && data.msgBox[data.selectedWindow])
data.msgBox[data.selectedWindow].hide() data.msgBox[data.selectedWindow].hide()
data.selectedWindow = boxname; if(boxname === undefined){
if(data.selectedWindow != data.statusWindow){ // Leave the clear
data.cmdline.setLabel('to '+data.selectedWindow)
data.markAsRead(data.selectedWindow)
} else { } else {
data.cmdline.setLabel('Command for Telecommander') // Switch window
data.selectedWindow = boxname;
if(data.selectedWindow != data.statusWindow){
data.cmdline.setLabel('to '+data.selectedWindow)
data.markAsRead(data.selectedWindow)
} else {
data.cmdline.setLabel('Command for Telecommander')
}
var newb = data.getMsgBox(data.selectedWindow)
newb.show()
newb.emit('scroll')
} }
var newb = data.getMsgBox(data.selectedWindow)
newb.show()
newb.emit('scroll')
data.screen.render() data.screen.render()
} }
@ -88,6 +94,20 @@ module.exports = function(data){
}) })
data.screen.append(data.chats) data.screen.append(data.chats)
// Command line prompt
data.cmdline = blessed.textbox({
keys: false,
mouse: true,
label: 'Command for Telecommander',
bottom: 0,
left: 'center',
width: '100%',
height: 3,
border: { type: 'line' },
style: data.getDefaultStyle()
})
data.screen.append(data.cmdline);
// Function to create a log box // Function to create a log box
data.mkBox = function(txt){ data.mkBox = function(txt){
var b = ChatBox({ var b = ChatBox({
@ -125,40 +145,44 @@ module.exports = function(data){
return b return b
} }
// Command line prompt data.getDefaultPopupStyle = function(){
data.cmdline = blessed.textbox({ return {
keys: false, width: '30%',
mouse: true, key: true,
label: 'Command for Telecommander', height: 'shrink',
bottom: 0, left: 'center',
left: 'center', top: 'center',
width: '100%', align: 'center',
height: 3, valign: 'center',
border: { type: 'line' }, border: { type: 'line' },
style: data.getDefaultStyle() style: data.getDefaultStyle()
}) }
data.screen.append(data.cmdline); }
// Widget used to show loading windows // Widget used to show loading windows
data.loader = blessed.loading({ data.loader = blessed.Loading(data.getDefaultPopupStyle())
width: '30%',
height: 5,
left: 'center',
top: 'center',
align: 'center',
valign: 'center',
border: { type: 'line' },
style: data.getDefaultStyle()
})
data.screen.append(data.loader) data.screen.append(data.loader)
data.load = function(msg){
data.loader.stop()
data.loader.load(msg)
}
// Widget used to ask for phone number and code
data.promptBox = blessed.Prompt(data.getDefaultPopupStyle())
data.screen.append(data.promptBox)
// Widget used to show pop up read only messages
data.popup = blessed.Message(data.getDefaultPopupStyle())
data.screen.append(data.popup)
data.popup.hide()
// mgsBox holds the chat window instance for every chat // mgsBox holds the chat window instance for every chat
data.msgBox = { } data.msgBox = { }
// Add the status window and switch to it // Add the status window but don't show it
data.msgBox[data.statusWindow] = data.mkBox(data.statusWindow) data.msgBox[data.statusWindow] = data.mkBox(data.statusWindow)
data.screen.append(data.msgBox[data.statusWindow]) data.screen.append(data.msgBox[data.statusWindow])
data.switchToBox(data.statusWindow) data.switchToBox()
data.screen.on('resize',function(){ data.screen.on('resize',function(){
for(i in data.msgBox){ for(i in data.msgBox){
@ -175,72 +199,7 @@ module.exports = function(data){
data.switchToBox(data.statusWindow) data.switchToBox(data.statusWindow)
}) })
data.command = function(cmd){ data.command = function(cmd){
cmdl = cmd.split(' ') data.log('Commands are not implemented... sorry!')
cmdname = cmdl[0]
if(cmdname === 'phone'){ // So the user can provide his phone numbah
if(data.connected){
return log("Silly user, you're already connected! We don't need that phone number")
}
data.user.phone = cmd.split(' ')[1]
var mindate = moment()
log('Checking your phone number with Telegram...')
data.client.auth.sendCode(user.phone,5,'en',function(result){
if(result.err_code){
return log('Errors:',result.error_code,result.error_message)
}
//log('Res:',JSON.stringify(result))
data.user.registered = result.phone_registered
data.user.phoneCodeHash = result.phone_code_hash
function gmd(){
var m = moment()
m = m.subtract(m.diff(mindate))
return 'Please use a telegram code not older than '+m.fromNow(true)
}
if(!data.user.registered){
data.log("Your number is not registered. Telecommander will register your account with the Telegram service")
data.log(gmd())
data.log('Ready for phone code, use command: "code <code> <name> <lastname>" to register')
data.log("If you don't want to sign up, just don't enter the code and press ESC to exit. No data was saved to the file system")
} else {
data.log("Your number is already assigned to a Telegram account. Telecommander will log you in.")
data.log(gmd())
data.log("If you don't want to sign in, just don't enter the code and press ESC to exit. No data was saved to the file system")
}
})
} else if(cmdname === 'code'){ // So the user can provide his phone code
if(data.connected){
return log("Silly user, you're already connected! We don't need that phone code")
}
code = cmdl[1]
name = cmdl[2]
lastname = cmdl[3]
if(((!name || !lastname) && !data.user.registered) || !code)
return log('insufficient arguments:',cmd)
cb = function(result){
data.user.id = ''+result.user.id
data.user.phone = result.user.phone
data.user.phoneCodeHash = result.phone_code_hash
data.user.username = result.user.username
data.user.first_name = result.user.first_name
data.user.last_name = result.user.last_name
// Done, write user data and key to disk
log('Writing Log In token and user data to',cfgDir)
fs.writeFile(cfgDir+'key',authKey,function(err){
if(err) log('FATAL: Could not write key to disk:',err)
})
fs.writeFile(cfgDir+'user_data.json',JSON.stringify(user),function(err){
if(err) log("FATAL: couldn't write user_data.json:",err)
})
whenReady()
}
// Log in finally
if(user.registered) client.auth.signIn(user.phone,user.phoneCodeHash,code,cb)
else client.auth.signUp(user.phone,user.phoneCodeHash,code,name,lastname,cb)
} else {
data.log('Command not found.')
}
} }
// What happens when a different window is selected // What happens when a different window is selected

View File

@ -127,18 +127,15 @@ module.exports = function(data){
toread = data.contacts[id].toread toread = data.contacts[id].toread
} else data.log('Failed to find name for',type,id) } else data.log('Failed to find name for',type,id)
if(obj === undefined){ if(obj === undefined){
throw new Error("Can't find name for",type,id) name = 'Unknown '+type+' '+id
return } else if(type === 'user'){
}
if(type === 'user'){
// User // User
if(obj.first_name === undefined && obj.last_name === undefined && obj.username === undefined){ if(obj.first_name === undefined && obj.last_name === undefined && obj.username === undefined){
data.log('Zombie User: '+obj) data.log('Zombie User: '+obj)
return 'undefined' return 'undefined'
} }
name = obj.first_name + (obj.last_name?' '+obj.last_name:'') + (obj.username?' (@'+obj.username+')':'') name = obj.first_name + (obj.last_name?' '+obj.last_name:'') + (obj.username?' (@'+obj.username+')':'')
} else { } else { // Group
// Group
name = obj.title name = obj.title
} }
if(forChatList && toread > 0) return '* '+name if(forChatList && toread > 0) return '* '+name

View File

@ -57,70 +57,13 @@ data.command = function(cmd){
if(data.connected){ if(data.connected){
return log("Silly user, you're already connected! We don't need that phone number") return log("Silly user, you're already connected! We don't need that phone number")
} }
data.user.phone = cmd.split(' ')[1]
var mindate = moment()
data.log('Checking your phone number with Telegram...')
data.client.auth.sendCode(data.user.phone,5,'en',function(result){
if(result.err_code){
return data.log('Errors:',result.error_code,result.error_message)
}
//data.log('Res:',JSON.stringify(result))
data.user.registered = result.phone_registered
data.user.phoneCodeHash = result.phone_code_hash
function gmd(){
var m = moment()
m = m.subtract(m.diff(mindate))
return 'Please use a telegram code not older than '+m.fromNow(true)
}
if(!data.user.registered){
data.log("Your number is not registered. Telecommander will register your account with the Telegram service")
data.log(gmd())
data.log('Ready for phone code, use command: "code <code> <name> <lastname>" to register')
data.log("If you don't want to sign up, just don't enter the code and press ESC to exit. No data was saved to the file system")
} else {
data.log("Your number is already assigned to a Telegram account. Telecommander will log you in.")
data.log(gmd())
data.log("If you don't want to sign in, just don't enter the code and press ESC to exit. No data was saved to the file system")
}
})
} else if(cmdname === 'code'){ // So the user can provide his phone code } else if(cmdname === 'code'){ // So the user can provide his phone code
if(data.connected){ if(data.connected){
return data.log("Silly user, you're already connected! We don't need that phone code") return data.log("Silly user, you're already connected! We don't need that phone code")
} }
code = cmdl[1]
name = cmdl[2]
lastname = cmdl[3]
if(((!name || !lastname) && !data.user.registered) || !code)
return log('insufficient arguments:',cmd)
var cb = function(result){
data.user.id = ''+result.user.id
data.user.phone = result.user.phone
data.user.phoneCodeHash = result.phone_code_hash
data.user.username = result.user.username
data.user.first_name = result.user.first_name
data.user.last_name = result.user.last_name
// Done, write user data and key to disk
try {
fs.mkdirSync(data.cfgDir,'0770')
} catch (e) {
if(e.code != 'EEXIST'){
console.error("FATAL: couldn't create configuration directory",data.cfgDir,e)
process.exit(-1)
}
}
data.log('Writing Log In token and user data to',data.cfgDir)
fs.writeFile(data.cfgDir+'key',data.app.authKey,function(err){
if(err) data.log('FATAL: Could not write key to disk:',err)
})
fs.writeFile(data.cfgDir+'user_data.json',JSON.stringify(data.user),function(err){
if(err) data.log("FATAL: couldn't write user_data.json:",err)
})
data.whenReady()
}
// Log in finally
if(data.user.registered) data.client.auth.signIn(data.user.phone,data.user.phoneCodeHash,code,cb)
else data.client.auth.signUp(data.user.phone,data.user.phoneCodeHash,code,name,lastname,cb)
} else { } else {
data.log('Command not found.') data.log('Command not found.')
} }
@ -142,15 +85,86 @@ data.sendMsg = function(name,str){
}) })
} }
data.onPhoneCode = function(something,s){
if(s === null){ // User cancelled
process.exit(0)
}
var cmdl = s.split(' ')
code = cmdl[0]
name = cmdl[1]
lastname = cmdl[2]
if(((!name || !lastname) && !data.user.registered) || !code)
return log('insufficient arguments:',cmd) // TODO: handle this better!
var cb = function(result){
data.user.id = ''+result.user.id
data.user.phone = result.user.phone
data.user.phoneCodeHash = result.phone_code_hash
data.user.username = result.user.username
data.user.first_name = result.user.first_name
data.user.last_name = result.user.last_name
// Done, write user data and key to disk
try {
fs.mkdirSync(data.cfgDir,'0770')
} catch (e) {
if(e.code != 'EEXIST'){
console.error("FATAL: couldn't create configuration directory",data.cfgDir,e)
process.exit(-1)
}
}
data.log('Writing Log In token and user data to',data.cfgDir)
fs.writeFile(data.cfgDir+'key',data.app.authKey,function(err){
if(err) data.log('FATAL: Could not write key to disk:',err)
})
fs.writeFile(data.cfgDir+'user_data.json',JSON.stringify(data.user),function(err){
if(err) data.log("FATAL: couldn't write user_data.json:",err)
})
data.whenReady()
}
// Log in finally
if(data.user.registered) data.client.auth.signIn(data.user.phone,data.user.phoneCodeHash,code,cb)
else data.client.auth.signUp(data.user.phone,data.user.phoneCodeHash,code,name,lastname,cb)
}
data.onPhoneNumber = function(something,s){
if(s === null){ // User cancelled
process.exit(0)
}
data.user.phone = s.trim()
var mindate = moment()
data.log('Checking your phone number with Telegram...')
data.client.auth.sendCode(data.user.phone,5,'en',function(result){
if(result.err_code){
return data.log('Errors:',result.error_code,result.error_message)
}
//data.log('Res:',JSON.stringify(result))
data.user.registered = result.phone_registered
data.user.phoneCodeHash = result.phone_code_hash
var msg
if(!data.user.registered){
msg = "Your number ("+data.user.phone+") is not registered.\nTelecommander will register your account with the Telegram service."
} else {
msg = "Your number ("+data.user.phone+") is already assigned to a Telegram account.\nTelecommander will log you in."
}
msg += "\nPress ESC to exit now, or enter to continue"
data.popup.display(msg,0,function(){
data.popup.hide()
data.promptBox.input('Your telegram code:','',data.onPhoneCode)
})
})
}
// Connects to telegram // Connects to telegram
data.connect = function(){ data.connect = function(){
data.load('Connecting...')
data.client = data.telegramLink.createClient(data.app, data.dataCenter, function(){ data.client = data.telegramLink.createClient(data.app, data.dataCenter, function(){
if(!data.app.authKey){ if(!data.app.authKey){
data.log('Downloading Authorization Key...') data.log('Downloading Authorization Key...')
data.client.createAuthKey(function(auth){ data.client.createAuthKey(function(auth){
data.app.authKey = auth.key.encrypt('password') // Will add security later, I promise data.app.authKey = auth.key.encrypt('password') // Will add security later, I promise
// Writes the new encrypted key to disk // Writes the new encrypted key to disk
data.log('Ready for phone number, use command: phone <number>') data.loader.stop()
//data.log('Ready for phone number, use command: phone <number>')
data.promptBox.input('Phone number (international format):','+',data.onPhoneNumber)
}) })
} else { } else {
data.whenReady() data.whenReady()
@ -164,7 +178,7 @@ data.connect = function(){
// Executed when connected and logged in // Executed when connected and logged in
data.whenReady = function(){ data.whenReady = function(){
data.log('Connected') data.load('Connected')
data.connected = true data.connected = true
data.downloadData() data.downloadData()
data.chats.focus() data.chats.focus()
@ -172,8 +186,7 @@ data.whenReady = function(){
// Downloads stuff // Downloads stuff
data.downloadData = function(){ data.downloadData = function(){
data.loader.load('Downloading data') data.load('Downloading data')
data.screen.render()
data.client.contacts.getContacts('',function(cont){ data.client.contacts.getContacts('',function(cont){
//data.chats.clearItems() //data.chats.clearItems()
//data.chats.add(data.statusWindow) //data.chats.add(data.statusWindow)
@ -184,6 +197,7 @@ data.downloadData = function(){
data.client.messages.getDialogs(0,0,10,function(dialogs){ data.client.messages.getDialogs(0,0,10,function(dialogs){
if(dialogs && dialogs.chats && dialogs.chats.list) if(dialogs && dialogs.chats && dialogs.chats.list)
dialogs.chats.list.forEach(data.addGroup) dialogs.chats.list.forEach(data.addGroup)
data.loader.stop()
}) })
data.client.updates.getState(function(astate){ data.client.updates.getState(function(astate){
@ -236,6 +250,7 @@ data.getMessages = function(name,box){
if(!peer) return log('Could not find peer:',name) if(!peer) return log('Could not find peer:',name)
data.downloadingMessages = true data.downloadingMessages = true
var oldnlines = box.getLines().length var oldnlines = box.getLines().length
if(data.selectedWindow === name) data.load('Downloading history...')
data.client.messages.getHistory(peer,0,obj.oldest_message||0,10,function(res){ data.client.messages.getHistory(peer,0,obj.oldest_message||0,10,function(res){
//log(res.toPrintable()) //log(res.toPrintable())
//log('Got history for: '+getName(peer.user_id||peer.chat_id,peer.chat_id?'group':'user')) //log('Got history for: '+getName(peer.user_id||peer.chat_id,peer.chat_id?'group':'user'))
@ -251,6 +266,7 @@ data.getMessages = function(name,box){
}) })
if(oldnlines == 0) box.setScrollPerc(100) if(oldnlines == 0) box.setScrollPerc(100)
//box.add(obj.oldest_message) //box.add(obj.oldest_message)
data.loader.stop()
data.downloadingMessages = false data.downloadingMessages = false
}) })
} }
@ -308,20 +324,25 @@ data.appendMsg = function(msg,toBoxId,bare,prepend){
box.add(msg) box.add(msg)
else { else {
var id = msg.from_id var id = msg.from_id
var date = moment.unix(msg.date).format('DD-MM-YYYY H:mm') if(!id){ // Weird zombie message!
name = data.getName(id,'user') data.log('Zombie Message:',msg.toPrintable())
var txt = (name || id)+' {|} {grey-fg}'+date+'{/grey-fg}\n' return box
if(msg.media){ } else { // Regular message
if(msg.media.photo) var date = moment.unix(msg.date).format('DD-MM-YYYY H:mm')
txt += '{grey-fg}>>>{/grey-fg} (Photo)' name = data.getName(id,'user')
else if(msg.media.audio) var txt = (name || id)+' {|} {grey-fg}'+date+'{/grey-fg}\n'
txt += "{grey-fg}>>>{/grey-fg} (Audio Message) "+msg.media.audio.duration+" seconds" if(msg.media){
else if(!msg.message) if(msg.media.photo)
txt += "{grey-fg}>>>{/grey-fg} (Unsupported Message)" txt += '{grey-fg}>>>{/grey-fg} (Photo)'
else if(msg.media.audio)
txt += "{grey-fg}>>>{/grey-fg} (Audio Message) "+msg.media.audio.duration+" seconds"
else if(!msg.message)
txt += "{grey-fg}>>>{/grey-fg} (Unsupported Message)"
}
if(msg.message) txt += '{grey-fg}>{/grey-fg} '+msg.message
if(prepend) box.prepend(txt)
else box.add(txt)
} }
if(msg.message) txt += '{grey-fg}>{/grey-fg} '+msg.message
if(prepend) box.prepend(txt)
else box.add(txt)
} }
// Mark messages as read if needed // Mark messages as read if needed
if(param === data.selectedWindow) data.markAsRead(param) if(param === data.selectedWindow) data.markAsRead(param)
@ -330,8 +351,8 @@ data.appendMsg = function(msg,toBoxId,bare,prepend){
// - Entry Point - // - Entry Point -
// Load authKey and userdata from disk, then act depending on outcome // Load authKey and userdata from disk, then act depending on outcome
data.load('Starting up...')
data.screen.render() data.screen.render()
data.log('Loading files...')
fs.exists(data.keyFile,function(exists){ fs.exists(data.keyFile,function(exists){
if(exists){ if(exists){
//log('Authorization Key found') //log('Authorization Key found')