gajim-plural/src/message_window.py

535 lines
17 KiB
Python
Raw Normal View History

## message_window.py
##
## Copyright (C) 2005-2006 Travis Shirk <travis@pobox.com>
##
## This program 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 2 only.
##
## This program 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.
##
import gtk
import gtk.glade
import pango
import gobject
import common
import gtkgui_helpers
import message_control
from common import gajim
####################
# FIXME: Can't this stuff happen once?
from common import i18n
_ = i18n._
APP = i18n.APP
GTKGUI_GLADE = 'gtkgui.glade'
####################
class MessageWindow:
'''Class for windows which contain message like things; chats,
groupchats, etc.'''
def __init__(self):
self._controls = {}
self.widget_name = 'message_window'
self.xml = gtk.glade.XML(GTKGUI_GLADE, self.widget_name, APP)
self.window = self.xml.get_widget(self.widget_name)
self.alignment = self.xml.get_widget('alignment')
self.notebook = self.xml.get_widget('notebook')
# Remove the glade pages
while self.notebook.get_n_pages():
self.notebook.remove_page(0)
# Tab customizations
pref_pos = gajim.config.get('tabs_position')
if pref_pos != 'top':
if pref_pos == 'bottom':
nb_pos = gtk.POS_BOTTOM
elif pref_pos == 'left':
nb_pos = gtk.POS_LEFT
elif pref_pos == 'right':
nb_pos = gtk.POS_RIGHT
else:
nb_pos = gtk.POS_TOP
else:
nb_pos = gtk.POS_TOP
self.notebook.set_tab_pos(nb_pos)
if gajim.config.get('tabs_always_visible'):
self.notebook.set_show_tabs(True)
self.alignment.set_property('top-padding', 2)
else:
self.notebook.set_show_tabs(False)
self.notebook.set_show_border(gajim.config.get('tabs_border'))
2006-01-02 10:04:30 +01:00
self.notebook.connect('switch-page',
self._on_notebook_switch_page)
# Connect event handling for this Window
self.window.connect('delete-event', self._on_window_delete)
self.window.connect('destroy', self._on_window_destroy)
self.window.connect('focus-in-event', self._on_window_focus)
# Restore previous window position
if gajim.config.get('saveposition'):
# get window position and size from config
gtkgui_helpers.move_window(self.window,
gajim.config.get('msgwin-x-position'),
gajim.config.get('msgwin-y-position'))
gtkgui_helpers.resize_window(self.window,
gajim.config.get('msgwin-width'),
gajim.config.get('msgwin-height'))
self.window.show_all()
def _on_window_focus(self, widget, event):
# FIXME:
print "_on_window_focus"
# window received focus, so if we had urgency REMOVE IT
# NOTE: we do not have to read the message (it maybe in a bg tab)
# to remove urgency hint so this functions does that
if gtk.gtk_version >= (2, 8, 0) and gtk.pygtk_version >= (2, 8, 0):
if widget.props.urgency_hint:
widget.props.urgency_hint = False
ctl = self.get_active_control()
2006-01-02 10:04:30 +01:00
if ctl:
ctl.set_control_active(True)
# Undo "unread" state display, etc.
if ctl.type_id == message_control.TYPE_GC:
self.redraw_tab(ctl.contact, 'active')
else:
# NOTE: we do not send any chatstate to preserve
# inactive, gone, etc.
self.redraw_tab(ctl.contact)
def _on_window_delete(self, win, event):
2006-01-01 20:40:05 +01:00
# Make sure all controls are okay with being deleted
for ctl in self._controls.values():
2006-01-02 10:04:30 +01:00
if not ctl.allow_shutdown():
2006-01-01 20:40:05 +01:00
return True # halt the delete
# FIXME: Do based on type, main, never, peracct, pertype
if gajim.config.get('saveposition'):
# save the window size and position
x, y = win.get_position()
gajim.config.set('msgwin-x-position', x)
gajim.config.set('msgwin-y-position', y)
width, height = win.get_size()
gajim.config.set('msgwin-width', width)
gajim.config.set('msgwin-height', height)
return False
2006-01-01 20:40:05 +01:00
def _on_window_destroy(self, win):
2006-01-01 20:40:05 +01:00
# FIXME
print "MessageWindow._on_window_destroy:", win
2006-01-01 20:40:05 +01:00
for ctl in self._controls.values():
ctl.shutdown()
self._controls.clear()
def new_tab(self, control):
assert(not self._controls.has_key(control.contact.jid))
self._controls[control.contact.jid] = control
2006-01-02 10:04:30 +01:00
if len(self._controls) > 1:
self.notebook.set_show_tabs(True)
self.alignment.set_property('top-padding', 2)
2006-01-02 10:04:30 +01:00
# Connect to keyboard events
control.widget.connect('key_press_event',
self.on_conversation_textview_key_press_event)
# FIXME: need to get this event without access to message_textvier
#control.widget.connect('mykeypress',
# self.on_message_textview_mykeypress_event)
control.widget.connect('key_press_event',
self.on_message_textview_key_press_event)
# Add notebook page and connect up to the tab's close button
xml = gtk.glade.XML(GTKGUI_GLADE, 'chat_tab_ebox', APP)
tab_label_box = xml.get_widget('chat_tab_ebox')
xml.signal_connect('on_close_button_clicked', self.on_close_button_clicked,
control.contact)
2006-01-02 10:04:30 +01:00
xml.signal_connect('on_tab_eventbox_button_press_event',
self.on_tab_eventbox_button_press_event, control.widget)
self.notebook.append_page(control.widget, tab_label_box)
2006-01-02 10:04:30 +01:00
self.redraw_tab(control.contact)
self.show_title()
self.window.show_all()
2006-01-02 10:04:30 +01:00
# NOTE: we do not call set_control_active(True) since we don't know whether
# the tab is the active one.
def on_tab_eventbox_button_press_event(self, widget, event, child):
if event.button == 3:
n = self.notebook.page_num(child)
self.notebook.set_current_page(n)
self.popup_menu(event)
def on_message_textview_mykeypress_event(self, widget, event_keyval,
event_keymod):
# FIXME: Not called yet
print "MessageWindow.on_message_textview_mykeypress_event:", event
# NOTE: handles mykeypress which is custom signal; see message_textview.py
# construct event instance from binding
event = gtk.gdk.Event(gtk.gdk.KEY_PRESS) # it's always a key-press here
event.keyval = event_keyval
event.state = event_keymod
event.time = 0 # assign current time
if event.keyval == gtk.keysyms.ISO_Left_Tab: # SHIFT + TAB
if event.state & gtk.gdk.CONTROL_MASK: # CTRL + SHIFT + TAB
self.notebook.emit('key_press_event', event)
if event.keyval == gtk.keysyms.Tab:
if event.state & gtk.gdk.CONTROL_MASK: # CTRL + TAB
self.notebook.emit('key_press_event', event)
def on_message_textview_key_press_event(self, widget, event):
print "MessageWindow.on_message_textview_key_press_event:", event
if event.keyval == gtk.keysyms.Page_Down: # PAGE DOWN
if event.state & gtk.gdk.CONTROL_MASK: # CTRL + PAGE DOWN
self.notebook.emit('key_press_event', event)
elif event.keyval == gtk.keysyms.Page_Up: # PAGE UP
if event.state & gtk.gdk.CONTROL_MASK: # CTRL + PAGE UP
self.notebook.emit('key_press_event', event)
def on_conversation_textview_key_press_event(self, widget, event):
'''Do not block these events and send them to the notebook'''
print "MessageWindow.on_conversation_textview_key_press_event:", event
if event.state & gtk.gdk.CONTROL_MASK:
if event.keyval == gtk.keysyms.Tab: # CTRL + TAB
self.notebook.emit('key_press_event', event)
elif event.keyval == gtk.keysyms.ISO_Left_Tab: # CTRL + SHIFT + TAB
self.notebook.emit('key_press_event', event)
elif event.keyval == gtk.keysyms.Page_Down: # CTRL + PAGE DOWN
self.notebook.emit('key_press_event', event)
elif event.keyval == gtk.keysyms.Page_Up: # CTRL + PAGE UP
self.notebook.emit('key_press_event', event)
def on_close_button_clicked(self, button, contact):
'''When close button is pressed: close a tab'''
self.remove_tab(contact)
def show_title(self, urgent = True):
'''redraw the window's title'''
unread = 0
for ctl in self._controls.values():
unread += ctl.nb_unread
start = ''
if unread > 1:
start = '[' + unicode(unread) + '] '
elif unread == 1:
start = '* '
ctl = self.get_active_control()
if len(self._controls) > 1: # if more than one tab in the same window
add = ctl.display_name
elif len(self._controls) == 1: # just one tab
add = ctl.contact.name
# FIXME: This is for GC only
# elif self.widget_name == 'groupchat_window':
# name = gajim.get_nick_from_jid(jid)
# add = name
title = start + add
if len(gajim.connections) >= 2: # if we have 2 or more accounts
title += ' (' + _('account: ') + ctl.account + ')'
# Update UI
self.window.set_title(title)
if urgent:
gtkgui_helpers.set_unset_urgency_hint(self.window, unread)
def set_active_tab(self, jid):
ctl = self._controls[jid]
ctl_page = self.notebook.page_num(ctl.widget)
self.notebook.set_current_page(ctl_page)
def remove_tab(self, contact):
2006-01-02 10:04:30 +01:00
ctl = self.get_control(contact.jid)
if len(self._controls) == 1 or not ctl.allow_shutdown():
return
if gajim.interface.systray_enabled:
gajim.interface.systray.remove_jid(contact.jid, ctl.account,
ctl.type)
ctl.shutdown()
self.notebook.remove_page(self.notebook.page_num(self.childs[jid]))
del self._controls[contact.jid]
del gajim.last_message_time[self.account][jid]
if len(self.xmls) == 1: # we now have only one tab
show_tabs_if_one_tab = gajim.config.get('tabs_always_visible')
self.notebook.set_show_tabs(show_tabs_if_one_tab)
if not show_tabs_if_one_tab:
self.alignment.set_property('top-padding', 0)
self.show_title()
2005-12-31 08:19:43 +01:00
def redraw_tab(self, contact, chatstate = None):
ctl = self._controls[contact.jid]
ctl.update_state()
hbox = self.notebook.get_tab_label(ctl.widget).get_children()[0]
status_img = hbox.get_children()[0]
nick_label = hbox.get_children()[1]
# Optionally hide close button
close_button = hbox.get_children()[2]
if gajim.config.get('tabs_close_button'):
close_button.show()
else:
close_button.hide()
# Update nick
2005-12-31 08:19:43 +01:00
nick_label.set_max_width_chars(10)
(tab_label_str, tab_label_color) = ctl.markup_tab_label(contact.name,
chatstate)
nick_label.set_markup(tab_label_str)
if tab_label_color:
nick_label.modify_fg(gtk.STATE_NORMAL, tab_label_color)
nick_label.modify_fg(gtk.STATE_ACTIVE, tab_label_color)
num_unread = ctl.nb_unread
# Set tab image (always 16x16); unread messages show the 'message' image
img_16 = gajim.interface.roster.get_appropriate_state_images(contact.jid)
if num_unread and gajim.config.get('show_unread_tab_icon'):
tab_img = img_16['message']
else:
tab_img = img_16[contact.show]
if tab_img.get_storage_type() == gtk.IMAGE_ANIMATION:
status_img.set_from_animation(tab_img.get_animation())
else:
status_img.set_from_pixbuf(tab_img.get_pixbuf())
def repaint_themed_widgets(self):
'''Repaint controls in the window with theme color'''
# iterate through controls and repaint
for ctl in self._controls.values():
ctl.repaint_themed_widgets()
2006-01-02 10:04:30 +01:00
def _widget_to_control(self, widget):
2005-12-31 04:53:48 +01:00
for ctl in self._controls.values():
if ctl.widget == widget:
return ctl
return None
def get_active_control(self):
2005-12-31 04:53:48 +01:00
notebook = self.notebook
active_widget = notebook.get_nth_page(notebook.get_current_page())
2006-01-02 10:04:30 +01:00
return self._widget_to_control(active_widget)
def get_active_contact(self):
return self.get_active_control().contact
2005-12-31 04:53:48 +01:00
def get_active_jid(self):
return self.get_active_contact().jid
def is_active(self):
return self.window.is_active()
def get_origin(self):
return self.window.window.get_origin()
2005-12-31 04:53:48 +01:00
def toggle_emoticons(self):
for ctl in self._controls.values():
ctl.toggle_emoticons()
def update_font(self):
for ctl in self._controls.values():
ctl.update_font()
def update_tags(self):
for ctl in self._controls.values():
ctl.update_tags()
2005-12-31 04:53:48 +01:00
2006-01-01 20:40:05 +01:00
def get_control(self, arg):
'''Return the MessageControl for jid or n, where n is the notebook page index'''
if isinstance(arg, unicode):
jid = arg
for ctl in self._controls.values():
if ctl.contact.jid == jid:
return ctl
return None
else:
page_num = arg
notebook = self.notebook
if page_num == None:
page_num = notebook.get_current_page()
nth_child = notebook.get_nth_page(page_num)
2006-01-02 10:04:30 +01:00
return self._widget_to_control(nth_child)
def controls(self):
for ctl in self._controls.values():
yield ctl
def update_print_time(self):
if gajim.config.get('print_time') != 'sometimes':
for ctl in self.controls():
if ctl.print_time_timeout_id:
gobject.source_remove(ctl.print_time_timeout_id)
del ctl.print_time_timeout_id
else:
for ctl in self.controls():
if not ctl.print_time_timeout_id:
ctl.print_time_timeout()
ctl.print_time_timeout_id = gobject.timeout_add(300000,
ctl.print_time_timeout, None)
2006-01-01 20:40:05 +01:00
def move_to_next_unread_tab(self, forward):
ind = self.notebook.get_current_page()
current = ind
found = False
# loop until finding an unread tab or having done a complete cycle
while True:
if forward == True: # look for the first unread tab on the right
ind = ind + 1
if ind >= self.notebook.get_n_pages():
ind = 0
else: # look for the first unread tab on the right
ind = ind - 1
if ind < 0:
ind = self.notebook.get_n_pages() - 1
if ind == current:
break # a complete cycle without finding an unread tab
ctl = self.get_control(ind)
if ctl.nb_unread > 0:
found = True
break # found
if found:
self.notebook.set_current_page(ind)
else: # not found
if forward: # CTRL + TAB
if current < (self.notebook.get_n_pages() - 1):
self.notebook.next_page()
else: # traverse for ever (eg. don't stop at last tab)
self.notebook.set_current_page(0)
else: # CTRL + SHIFT + TAB
if current > 0:
self.notebook.prev_page()
else: # traverse for ever (eg. don't stop at first tab)
self.notebook.set_current_page(
self.notebook.get_n_pages() - 1)
2006-01-02 10:04:30 +01:00
def popup_menu(self, event):
menu = self.get_active_control().prepare_context_menu()
# common menuitems (tab switches)
if len(self._controls) > 1: # if there is more than one tab
menu.append(gtk.SeparatorMenuItem()) # seperator
for ctl in self._controls.values():
jid = ctl.contact.jid
if jid != self.get_active_jid():
item = gtk.ImageMenuItem(_('Switch to %s') %\
self.names[jid])
img = gtk.image_new_from_stock(gtk.STOCK_JUMP_TO,
gtk.ICON_SIZE_MENU)
item.set_image(img)
item.connect('activate',
lambda obj, jid:self.set_active_tab(jid), jid)
menu.append(item)
# show the menu
menu.popup(None, None, None, event.button, event.time)
menu.show_all()
def _on_notebook_switch_page(self, notebook, page, page_num):
old_no = notebook.get_current_page()
old_ctl = self._widget_to_control(notebook.get_nth_page(old_no))
old_ctl.set_control_active(False)
new_ctl = self._widget_to_control(notebook.get_nth_page(page_num))
new_ctl.set_control_active(True)
################################################################################
class MessageWindowMgr:
'''A manager and factory for MessageWindow objects'''
# These constants map to common.config.opt_one_window_types indices
CONFIG_NEVER = 0
CONFIG_ALWAYS = 1
CONFIG_PERACCT = 2
CONFIG_PERTYPE = 3
# A key constant for the main window for all messages
MAIN_WIN = 'main'
def __init__(self):
''' A dictionary of windows; the key depends on the config:
CONFIG_NEVER: The key is the contact JID
CONFIG_ALWAYS: The key is MessageWindowMgr.MAIN_WIN
CONFIG_PERACCT: The key is the account name
CONFIG_PERTYPE: The key is a message type constant'''
self._windows = {}
# Map the mode to a int constant for frequent compares
mode = gajim.config.get('one_message_window')
self.mode = common.config.opt_one_window_types.index(mode)
assert(self.mode != -1)
def _new_window(self):
win = MessageWindow()
# we track the lifetime of this window
win.window.connect('destroy', self._on_window_destroy)
return win
def _gtkWinToMsgWin(self, gtk_win):
for w in self._windows.values():
if w.window == gtk_win:
return w
return None
def _on_window_destroy(self, win):
2006-01-01 20:40:05 +01:00
for k in self._windows.keys():
if self._windows[k].window == win:
del self._windows[k]
return
# How was the window not in out list?!? Assert.
assert(False)
def get_window(self, jid):
for win in self._windows.values():
2006-01-01 20:40:05 +01:00
if win.get_control(jid):
return win
return None
def has_window(self, jid):
return self.get_window(jid)
def create_window(self, contact, acct, type):
key = None
if self.mode == self.CONFIG_NEVER:
key = contact.jid
elif self.mode == self.CONFIG_ALWAYS:
key = self.MAIN_WIN
elif self.mode == self.CONFIG_PERACCT:
key = acct
elif self.mode == self.CONFIG_PERTYPE:
key = type
win = None
try:
win = self._windows[key]
except KeyError:
# FIXME
print "Creating tabbed chat window for '%s'" % str(key)
win = self._new_window()
self._windows[key] = win
assert(win)
return win
def get_control(self, jid):
2006-01-01 20:40:05 +01:00
'''Amonst all windows, return the MessageControl for jid'''
win = self.get_window(jid)
if win:
2006-01-01 20:40:05 +01:00
return win.get_control(jid)
return None
def windows(self):
for w in self._windows.values():
yield w
def controls(self):
for w in self._windows:
for c in w.controls():
yield c