poezio/src/gui.py

469 lines
18 KiB
Python
Raw Normal View History

2010-01-10 20:14:17 +00:00
#!/usr/bin/python
# -*- coding:utf-8 -*-
#
# Copyright 2010 Le Coz Florent <louizatakk@fedoraproject.org>
#
# This file is part of Poezio.
#
# Poezio 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, version 3 of the License.
#
# Poezio 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 Poezio. If not, see <http://www.gnu.org/licenses/>.
from handler import Handler
import curses
from curses import textpad
2010-01-13 01:04:30 +00:00
import locale
from datetime import datetime
2010-01-26 17:10:37 +00:00
from logging import logger
from random import randrange
from config import config
2010-01-13 01:04:30 +00:00
locale.setlocale(locale.LC_ALL, '')
code = locale.getpreferredencoding()
2010-01-12 11:52:26 +00:00
import sys
2010-01-12 14:57:58 +00:00
from connection import *
from window import Window
2010-01-12 14:57:58 +00:00
class User(object):
2010-01-12 14:16:13 +00:00
"""
keep trace of an user in a Room
2010-01-12 14:16:13 +00:00
"""
def __init__(self, nick, affiliation, show, status, role):
self.update(affiliation, show, status, role)
self.change_nick(nick)
self.color = randrange(2, 10)
2010-01-13 01:04:30 +00:00
def update(self, affiliation, show, status, role):
self.affiliation = None
self.show = None
self.status = status
self.role = role
2010-01-13 01:04:30 +00:00
def change_nick(self, nick):
self.nick = nick.encode('utf-8')
2010-01-13 01:04:30 +00:00
class Room(object):
2010-01-11 13:28:57 +00:00
"""
"""
def __init__(self, name, nick):
2010-01-11 13:28:57 +00:00
self.name = name
self.own_nick = nick
self.joined = False # false until self presence is received
self.users = []
self.lines = [] # (time, nick, msg) or (time, info)
self.topic = ''
2010-01-12 14:16:13 +00:00
2010-01-30 04:02:23 +00:00
def disconnect(self):
self.joined = False
self.users = []
def add_message(self, nick, msg):
if not msg:
logger.info('msg is None..., %s' % (nick))
return
self.lines.append((datetime.now(), nick.encode('utf-8'), msg.encode('utf-8')))
2010-01-12 14:16:13 +00:00
def add_info(self, info):
""" info, like join/quit/status messages"""
self.lines.append((datetime.now(), info.encode('utf-8')))
return info.encode('utf-8')
2010-01-11 13:28:57 +00:00
2010-01-30 04:02:23 +00:00
def get_user_by_name(self, nick):
fd = open('fion', 'w')
fd.write(nick)
# fd.write('Looking for %s\n' % nick)
for user in self.users:
# fd.write(user.nick)
if user.nick == nick:
return user
return None
def on_presence(self, stanza, nick):
2010-01-11 13:28:57 +00:00
"""
"""
affiliation = stanza.getAffiliation()
show = stanza.getShow()
status = stanza.getStatus()
role = stanza.getRole()
2010-01-30 04:02:23 +00:00
if not self.joined: # user in the room BEFORE us.
self.users.append(User(nick, affiliation, show, status, role))
2010-01-27 21:58:02 +00:00
if nick.encode('utf-8') == self.own_nick:
self.joined = True
return self.add_info("%s is in the room" % (nick))
change_nick = stanza.getStatusCode() == '303'
2010-01-28 00:55:02 +00:00
kick = stanza.getStatusCode() == '307'
2010-01-30 04:02:23 +00:00
user = self.get_user_by_name(nick)
# New user
if not user:
self.users.append(User(nick, affiliation, show, status, role))
return self.add_info('%s joined the room %s' % (nick, self.name))
# nick change
if change_nick:
if user.nick == self.own_nick:
self.own_nick = stanza.getNick().encode('utf-8')
user.change_nick(stanza.getNick())
return self.add_info('%s is now known as %s' % (nick, stanza.getNick()))
# kick
if kick:
self.users.remove(user)
reason = stanza.getReason().encode('utf-8') or ''
try:
by = stanza.getActor().encode('utf-8')
except:
by = None
if nick == self.own_nick:
self.disconnect()
if by:
return self.add_info('You have been kicked by %s. Reason: %s' % (by, reason))
else:
return self.add_info('You have been kicked. Reason: %s' % (reason))
else:
if by:
return self.add_info('%s has been kicked by %s. Reason: %s' % (nick, by, reason))
else:
return self.add_info('%s has been kicked. Reason: %s' % (nick, reason))
# user quit
if status == 'offline' or role == 'none':
self.users.remove(user)
return self.add_info('%s has left the room' % (nick))
# status change
user.update(affiliation, show, status, role)
return self.add_info('%s, status : %s, %s, %s, %s' % (nick, affiliation, role, show, status))
2010-01-11 13:28:57 +00:00
2010-01-10 20:14:17 +00:00
class Gui(object):
"""
Graphical user interface using ncurses
"""
def __init__(self, stdscr=None, muc=None):
self.init_curses(stdscr)
self.stdscr = stdscr
self.stdscr.leaveok(1)
self.rooms = [Room('Info', '')] # current_room is self.rooms[0]
self.window = Window(stdscr)
self.window.text_win.new_win('Info')
self.window.refresh(self.rooms[0])
self.muc = muc
2010-01-10 20:14:17 +00:00
2010-01-13 01:04:30 +00:00
self.commands = {
2010-01-30 04:42:29 +00:00
'help': (self.command_help, 'OLOL, this is SOOO recursive'),
'join': (self.command_join, 'Usage: /join [room_name][/nick]\nJoin: Join the specified room. You can specify a nickname after a slash (/). If no nickname is specified you will use the default_nick in the configuration file. You can omit the room name: you will then join the room you\'re looking at (useful if you were kicked). Examples:\n/join room@server.tld\n/join room@server.tld/John\n/join /me_again\n/join'),
'quit': (self.command_quit, 'Usage: /quit\nQuit: Just disconnect from the server and exit poezio.'),
'exit': (self.command_quit, 'Usage: /exit\nExit: Just disconnect from the server and exit poezio.'),
'next': (self.rotate_rooms_left, 'Usage: /next\nNext: Go to the next room.'),
'prev': (self.rotate_rooms_right, 'Usage: /prev\nPrev: Go to the previous room.'),
'part': (self.command_part, 'Usage: /part [message]\nPart: disconnect from a room. You can specify an optionnal message.'),
2010-01-31 03:07:30 +00:00
'show': (self.command_show, 'Usage: /show <availability> [status]\nShow: Change your availability and (optionnaly) your status. The <availability> argument is one of "avail, available, ok, here, chat, away, afk, dnd, busy, xa" and the optional [message] argument will be your status message'),
'away': (self.command_away, 'Usage: /away [message]\nAway: Sets your availability to away and (optional) sets your status message. This is equivalent do "/show away [message]"'),
'busy': (self.command_busy, 'Usage: /busy [message]\nBusy: Sets your availability to busy and (optional) sets your status message. This is equivalent do "/show busy [message]"'),
'avail': (self.command_avail, 'Usage: /avail [message]\nAvail: Sets your availability to available and (optional) sets your status message. This is equivalent do "/show available [message]"'),
'available': (self.command_avail, 'Usage: /available [message]\nAvailable: Sets your availability to available and (optional) sets your status message. This is equivalent do "/show available [message]"'),
2010-01-30 04:42:29 +00:00
'nick': (self.command_nick, 'Usage: /nick <nickname>\nNick: Change your nickname in the current room')
2010-01-13 01:04:30 +00:00
}
self.key_func = {
"KEY_LEFT": self.window.input.key_left,
"KEY_RIGHT": self.window.input.key_right,
"KEY_UP": self.window.input.key_up,
"KEY_END": self.window.input.key_end,
"KEY_HOME": self.window.input.key_home,
"KEY_DOWN": self.window.input.key_down,
2010-01-27 22:29:28 +00:00
"KEY_DC": self.window.input.key_dc,
2010-01-28 00:55:02 +00:00
"KEY_F(5)": self.rotate_rooms_left,
"KEY_F(6)": self.rotate_rooms_right,
"kLFT5": self.rotate_rooms_left,
"kRIT5": self.rotate_rooms_right,
"KEY_BACKSPACE": self.window.input.key_backspace
}
self.handler = Handler()
self.handler.connect('on-connected', self.on_connected)
self.handler.connect('join-room', self.join_room)
self.handler.connect('room-presence', self.room_presence)
self.handler.connect('room-message', self.room_message)
2010-01-29 16:24:44 +00:00
self.handler.connect('room-iq', self.room_iq)
def main_loop(self, stdscr):
while 1:
curses.doupdate()
2010-01-31 01:22:39 +00:00
try:
key = stdscr.getkey()
except:
self.window.resize(stdscr)
self.window.refresh(self.current_room())
if str(key) in self.key_func.keys():
self.key_func[key]()
2010-01-31 01:22:39 +00:00
elif str(key) == 'KEY_RESIZE':
self.window.resize(stdscr)
self.window.refresh(self.current_room())
elif len(key) >= 4:
continue
elif ord(key) == 10:
self.execute()
2010-01-27 22:32:20 +00:00
elif ord(key) == 8 or ord(key) == 127:
2010-01-27 22:29:28 +00:00
self.window.input.key_backspace()
elif ord(key) < 32:
continue
else:
2010-01-28 00:55:02 +00:00
if ord(key) == 27 and ord(stdscr.getkey()) == 91:
last = ord(stdscr.getkey()) # FIXME: ugly ugly workaroung.
if last == 51:
self.window.input.key_dc()
continue
2010-01-27 22:37:40 +00:00
elif ord(key) > 190 and ord(key) < 225:
key = key+stdscr.getkey()
elif ord(key) == 226:
key = key+stdscr.getkey()
key = key+stdscr.getkey()
self.window.do_command(key)
2010-01-26 17:10:37 +00:00
def current_room(self):
return self.rooms[0]
def get_room_by_name(self, name):
for room in self.rooms:
if room.name == name:
return room
return None
def init_curses(self, stdscr):
curses.start_color()
curses.noecho()
stdscr.keypad(True)
curses.init_pair(1, curses.COLOR_WHITE, curses.COLOR_BLUE)
curses.init_pair(2, curses.COLOR_BLUE, curses.COLOR_BLACK)
curses.init_pair(3, curses.COLOR_RED, curses.COLOR_BLACK) # Admin
curses.init_pair(4, curses.COLOR_BLUE, curses.COLOR_BLACK) # Participant
curses.init_pair(5, curses.COLOR_WHITE, curses.COLOR_BLACK) # Visitor
curses.init_pair(6, curses.COLOR_CYAN, curses.COLOR_BLACK)
curses.init_pair(7, curses.COLOR_GREEN, curses.COLOR_BLACK)
curses.init_pair(8, curses.COLOR_MAGENTA, curses.COLOR_BLACK)
curses.init_pair(9, curses.COLOR_YELLOW, curses.COLOR_BLACK)
2010-01-26 17:10:37 +00:00
def reset_curses(self):
curses.echo()
curses.endwin()
2010-01-26 17:10:37 +00:00
2010-01-29 16:24:44 +00:00
def on_connected(self, jid):
2010-01-28 00:55:02 +00:00
self.information("Welcome on Poezio \o/ !")
2010-01-29 16:24:44 +00:00
self.information("Your JID is %s" % jid)
pass
def join_room(self, room, nick):
self.window.text_win.new_win(room)
self.rooms.insert(0, Room(room, nick))
2010-01-26 17:10:37 +00:00
self.window.refresh(self.current_room())
2010-01-28 00:55:02 +00:00
def rotate_rooms_left(self, args=None):
self.rooms.append(self.rooms.pop(0))
2010-01-26 17:10:37 +00:00
self.window.refresh(self.current_room())
2010-01-28 00:55:02 +00:00
def rotate_rooms_right(self, args=None):
self.rooms.insert(0, self.rooms.pop())
2010-01-26 17:10:37 +00:00
self.window.refresh(self.current_room())
def room_message(self, stanza):
2010-01-29 16:24:44 +00:00
if len(sys.argv) > 1:
self.information(str(stanza))
if stanza.getType() != 'groupchat':
return # ignore all messages not comming from a MUC
room_from = stanza.getFrom().getStripped()
nick_from = stanza.getFrom().getResource()
if not nick_from:
nick_from = ''
room = self.get_room_by_name(room_from)
2010-01-26 17:10:37 +00:00
if not room:
2010-01-28 00:55:02 +00:00
self.information("message received for a non-existing room: %s" % (name))
return
body = stanza.getBody()
if not body:
body = stanza.getSubject()
2010-01-28 00:55:02 +00:00
info = room.add_info("%s changed the subject to: %s" % (nick_from, stanza.getSubject()))
self.window.text_win.add_line(room, (datetime.now(), info))
room.topic = stanza.getSubject().encode('utf-8').replace('\n', '|')
if room == self.current_room():
self.window.topic_win.refresh(room.topic)
curses.doupdate()
else:
room.add_message(nick_from, body)
self.window.text_win.add_line(room, (datetime.now(), nick_from.encode('utf-8'), body.encode('utf-8')))
2010-01-26 17:10:37 +00:00
if room == self.current_room():
self.window.text_win.refresh(room.name)
2010-01-26 17:10:37 +00:00
self.window.input.refresh()
2010-01-28 00:55:02 +00:00
curses.doupdate()
def room_presence(self, stanza):
2010-01-28 00:55:02 +00:00
if len(sys.argv) > 1:
self.information(str(stanza))
from_nick = stanza.getFrom().getResource()
from_room = stanza.getFrom().getStripped()
room = self.get_room_by_name(from_room)
2010-01-26 17:10:37 +00:00
if not room:
2010-01-29 16:24:44 +00:00
self.information("presence received for a non-existing room: %s" % (name))
2010-01-31 00:25:17 +00:00
if stanza.getType() == 'error':
msg = "Error: %s" % stanza.getError()
else:
msg = room.on_presence(stanza, from_nick)
if room == self.current_room():
self.window.text_win.add_line(room, (datetime.now(), msg))
self.window.text_win.refresh(room.name)
self.window.user_win.refresh(room.users)
2010-01-30 04:02:23 +00:00
self.window.text_win.refresh()
curses.doupdate()
2010-01-13 01:04:30 +00:00
2010-01-29 16:24:44 +00:00
def room_iq(self, iq):
if len(sys.argv) > 1:
self.information(str(iq))
2010-01-13 01:04:30 +00:00
def execute(self):
line = self.window.input.get_text()
self.window.input.clear_text()
2010-01-30 03:06:38 +00:00
self.window.input.refresh()
curses.doupdate()
if line == "":
return
if line.startswith('/'):
2010-01-13 01:04:30 +00:00
command = line.strip()[:].split()[0][1:]
args = line.strip()[:].split()[1:]
if command in self.commands.keys():
2010-01-30 04:42:29 +00:00
func = self.commands[command][0]
2010-01-13 01:04:30 +00:00
func(args)
2010-01-27 22:29:28 +00:00
return
2010-01-26 17:10:37 +00:00
if self.current_room().name != 'Info':
self.muc.send_message(self.current_room().name, line)
self.window.input.refresh()
2010-01-13 01:04:30 +00:00
2010-01-30 04:42:29 +00:00
def command_help(self, args):
room = self.current_room()
if len(args) == 0:
msg = 'Available commands are:'
for command in self.commands.keys():
msg += "%s " % command
msg += "\nType /help <command_name> to know what each command does"
if len(args) == 1:
if args[0] in self.commands.keys():
msg = self.commands[args[0]][1]
else:
msg = 'Unknown command : %s' % args[0]
room.add_info(msg)
self.window.text_win.add_line(room, (datetime.now(), msg))
self.window.text_win.refresh(room.name)
self.window.input.refresh()
2010-01-13 01:04:30 +00:00
def command_join(self, args):
2010-01-30 04:02:23 +00:00
if len(args) == 0:
r = self.current_room()
if r.name == 'Info':
return
room = r.name
nick = r.own_nick
else:
2010-01-30 04:02:23 +00:00
info = args[0].split('/')
if len(info) == 1:
nick = config.get('default_nick', 'Poezio')
else:
nick = info[1]
2010-01-31 03:07:30 +00:00
if info[0] == '': # happens with /join /nickname, which is OK
2010-01-30 04:02:23 +00:00
r = self.current_room()
if r.name == 'Info':
return
room = r.name
else:
room = info[0]
r = self.get_room_by_name(room)
if r and r.joined: # if we are already in the room
2010-01-28 00:55:02 +00:00
self.information("already in room [%s]" % room)
return
self.muc.join_room(room, nick)
2010-01-30 04:02:23 +00:00
if not r: # if the room window exists, we don't recreate it.
self.join_room(room, nick)
2010-01-13 01:04:30 +00:00
2010-01-31 00:25:17 +00:00
def command_show(self, args):
2010-01-31 03:07:30 +00:00
possible_show = {'avail':'None',
'available':'None',
'ok':'None',
'here':'None',
'chat':'chat',
'away':'away',
'afk':'away',
'dnd':'dnd',
'busy':'dnd',
'xa':'xa'
}
if len(args) < 1:
return
if not args[0] in possible_show.keys():
self.command_help(['show'])
return
show = possible_show[args[0]]
if len(args) > 1:
msg = ' '.join(args[1:])
else:
msg = None
for room in self.rooms:
if room.joined:
self.muc.change_show(room.name, room.own_nick, show, msg)
def command_away(self, args):
args.insert(0, 'away')
self.command_show(args)
def command_busy(self, args):
args.insert(0, 'busy')
self.command_show(args)
def command_avail(self, args):
args.insert(0, 'available')
self.command_show(args)
2010-01-31 00:25:17 +00:00
2010-01-30 04:02:23 +00:00
def command_part(self, args):
reason = None
room = self.current_room()
if room.name == 'Info':
return
if len(args):
msg = ' '.join(args)
else:
msg = None
2010-01-30 04:42:29 +00:00
if room.joined:
self.muc.quit_room(room.name, room.own_nick, msg)
2010-01-30 04:02:23 +00:00
self.rooms.remove(self.current_room())
self.window.refresh(self.current_room())
2010-01-30 03:06:38 +00:00
def command_nick(self, args):
if len(args) != 1:
return
nick = args[0]
room = self.current_room()
if not room.joined or room.name == "Info":
return
self.muc.change_nick(room.name, nick)
2010-01-28 00:55:02 +00:00
def information(self, msg):
room = self.get_room_by_name("Info")
info = room.add_info(msg)
if self.current_room() == room:
self.window.text_win.add_line(room, (datetime.now(), info))
2010-01-29 16:24:44 +00:00
self.window.text_win.refresh(room.name)
curses.doupdate()
2010-01-28 00:55:02 +00:00
2010-01-13 01:04:30 +00:00
def command_quit(self, args):
2010-01-26 17:10:37 +00:00
self.reset_curses()
2010-01-13 01:04:30 +00:00
sys.exit()