1383 lines
50 KiB
Python
1383 lines
50 KiB
Python
## plugins/tabbed_chat_window.py
|
|
##
|
|
## Gajim Team:
|
|
## - Yann Le Boulanger <asterix@lagaule.org>
|
|
## - Vincent Hanquez <tab@snarc.org>
|
|
## - Nikos Kouremenos <kourem@gmail.com>
|
|
##
|
|
## Copyright (C) 2003-2005 Gajim Team
|
|
##
|
|
## 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 time
|
|
import dialogs
|
|
import history_window
|
|
import gtkgui_helpers
|
|
|
|
try:
|
|
import gtkspell
|
|
except:
|
|
pass
|
|
|
|
from common import gajim
|
|
from common import helpers
|
|
from common import i18n
|
|
|
|
_ = i18n._
|
|
APP = i18n.APP
|
|
gtk.glade.bindtextdomain(APP, i18n.DIR)
|
|
gtk.glade.textdomain(APP)
|
|
|
|
GTKGUI_GLADE = 'gtkgui.glade'
|
|
|
|
class Chat:
|
|
"""Class for chat/groupchat windows"""
|
|
def __init__(self, plugin, account, widget_name):
|
|
self.xml = gtk.glade.XML(GTKGUI_GLADE, widget_name, APP)
|
|
self.window = self.xml.get_widget(widget_name)
|
|
|
|
self.widget_name = widget_name
|
|
|
|
self.plugin = plugin
|
|
self.account = account
|
|
self.change_cursor = None
|
|
self.xmls = {}
|
|
self.tagIn = {} # holds tag for nick that talks to us
|
|
self.tagOut = {} # holds tag for our nick
|
|
self.tagStatus = {} # holds status messages
|
|
self.nb_unread = {}
|
|
self.last_time_printout = {}
|
|
self.print_time_timeout_id = {}
|
|
self.names = {} # what is printed in the tab (eg. user.name)
|
|
self.childs = {} # holds the contents for every tab (VBox)
|
|
self.popup_is_shown = False # is a context menu shown or not?
|
|
|
|
# the following vars are used to keep history of user's messages
|
|
self.sent_history = {}
|
|
self.sent_history_pos = {}
|
|
self.typing_new = {}
|
|
self.orig_msg = {}
|
|
|
|
# we check that on opening new windows
|
|
self.always_compact_view = gajim.config.get('always_compact_view')
|
|
|
|
# notebook customizations
|
|
self.notebook = self.xml.get_widget('chat_notebook')
|
|
self.notebook.remove_page(0)
|
|
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)
|
|
self.notebook.set_show_tabs(gajim.config.get('tabs_always_visible'))
|
|
self.notebook.set_show_border(gajim.config.get('tabs_border'))
|
|
|
|
# muc attention states (when we are mentioned in a muc)
|
|
# if the room jid is in the list, the room has mentioned us
|
|
self.muc_attentions = []
|
|
|
|
def update_font(self):
|
|
font = pango.FontDescription(gajim.config.get('conversation_font'))
|
|
for jid in self.tagIn:
|
|
conversation_textview = self.xmls[jid].get_widget(
|
|
'conversation_textview')
|
|
conversation_textview.modify_font(font)
|
|
message_textview = self.xmls[jid].get_widget('message_textview')
|
|
message_textview.modify_font(font)
|
|
|
|
def update_tags(self):
|
|
for jid in self.tagIn:
|
|
self.tagIn[jid].set_property('foreground',
|
|
gajim.config.get('inmsgcolor'))
|
|
self.tagOut[jid].set_property('foreground',
|
|
gajim.config.get('outmsgcolor'))
|
|
self.tagStatus[jid].set_property('foreground',
|
|
gajim.config.get('statusmsgcolor'))
|
|
|
|
def update_print_time(self):
|
|
if gajim.config.get('print_time') != 'sometimes':
|
|
list_jid = self.print_time_timeout_id.keys()
|
|
for jid in list_jid:
|
|
gobject.source_remove(self.print_time_timeout_id[jid])
|
|
del self.print_time_timeout_id[jid]
|
|
else:
|
|
for jid in self.xmls:
|
|
if self.print_time_timeout_id.has_key(jid):
|
|
continue
|
|
self.print_time_timeout(jid)
|
|
self.print_time_timeout_id[jid] = \
|
|
gobject.timeout_add(300000,
|
|
self.print_time_timeout,
|
|
jid)
|
|
|
|
def show_title(self, urgent = True):
|
|
"""redraw the window's title"""
|
|
unread = 0
|
|
for jid in self.nb_unread:
|
|
unread += self.nb_unread[jid]
|
|
start = ""
|
|
if unread > 1:
|
|
start = '[' + unicode(unread) + '] '
|
|
elif unread == 1:
|
|
start = '* '
|
|
chat = self.names[jid]
|
|
if len(self.xmls) > 1: # if more than one tab in the same window
|
|
if self.widget_name == 'tabbed_chat_window':
|
|
add = _('Chat')
|
|
elif self.widget_name == 'groupchat_window':
|
|
add = _('Group Chat')
|
|
elif len(self.xmls) == 1: # just one tab
|
|
if self.widget_name == 'tabbed_chat_window':
|
|
c = gajim.get_first_contact_instance_from_jid(self.account, jid)
|
|
if c is None: # FIXME: I don't know why but c can be None!
|
|
add = ''
|
|
else:
|
|
add = c.name
|
|
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: ') + self.account + ')'
|
|
|
|
self.window.set_title(title)
|
|
if urgent:
|
|
gtkgui_helpers.set_unset_urgency_hint(self.window, unread)
|
|
|
|
def redraw_tab(self, jid, chatstate = None):
|
|
'''redraw the label of the tab
|
|
if chatstate is given that means we have HE SENT US a chatstate'''
|
|
# Update status images
|
|
self.set_state_image(jid)
|
|
|
|
child = self.childs[jid]
|
|
hb = self.notebook.get_tab_label(child).get_children()[0]
|
|
if self.widget_name == 'tabbed_chat_window':
|
|
nickname = hb.get_children()[1]
|
|
close_button = hb.get_children()[2]
|
|
|
|
unread = ''
|
|
num_unread = self.nb_unread[jid]
|
|
if num_unread == 1 and not gajim.config.get('show_unread_tab_icon'):
|
|
unread = '* '
|
|
elif num_unread > 1:
|
|
unread = '[' + unicode(num_unread) + '] '
|
|
|
|
# Draw tab label using chatstate
|
|
theme = gajim.config.get('roster_theme')
|
|
color = None
|
|
if unread and chatstate == 'active':
|
|
color = gajim.config.get_per('themes', theme,
|
|
'state_unread_color')
|
|
elif chatstate is not None:
|
|
if chatstate == 'composing':
|
|
color = gajim.config.get_per('themes', theme,
|
|
'state_composing_color')
|
|
elif chatstate == 'inactive':
|
|
color = gajim.config.get_per('themes', theme,
|
|
'state_inactive_color')
|
|
elif chatstate == 'gone':
|
|
color = gajim.config.get_per('themes', theme,
|
|
'state_gone_color')
|
|
elif chatstate == 'paused':
|
|
color = gajim.config.get_per('themes', theme,
|
|
'state_paused_color')
|
|
elif unread and self.window.get_property('has-toplevel-focus'):
|
|
color = gajim.config.get_per('themes', theme,
|
|
'state_active_color')
|
|
elif unread:
|
|
color = gajim.config.get_per('themes', theme,
|
|
'state_unread_color')
|
|
else:
|
|
color = gajim.config.get_per('themes', theme,
|
|
'state_active_color')
|
|
if color:
|
|
color = gtk.gdk.colormap_get_system().alloc_color(color)
|
|
# The widget state depend on whether this tab is the "current" tab
|
|
if self.notebook.page_num(child) == self.notebook.get_current_page():
|
|
widget_state = gtk.STATE_NORMAL
|
|
else:
|
|
widget_state = gtk.STATE_ACTIVE
|
|
if chatstate in ['inactive', 'gone']:
|
|
# Adjust color to be lighter against the darker inactive
|
|
# background
|
|
p = 0.4
|
|
mask = 0
|
|
color.red = int((color.red * p) + (mask * (1 - p)))
|
|
color.green = int((color.green * p) + (mask * (1 - p)))
|
|
color.blue = int((color.blue * p) + (mask * (1 - p)))
|
|
nickname.modify_fg(widget_state, color)
|
|
elif self.widget_name == 'groupchat_window':
|
|
nickname = hb.get_children()[0]
|
|
close_button = hb.get_children()[1]
|
|
|
|
unread = ''
|
|
has_focus = self.window.get_property('has-toplevel-focus')
|
|
current_tab = (self.notebook.page_num(child) == self.notebook.get_current_page())
|
|
color = None
|
|
theme = gajim.config.get('roster_theme')
|
|
if chatstate == 'attention' and (not has_focus or not current_tab):
|
|
if jid not in self.muc_attentions:
|
|
self.muc_attentions.append(jid)
|
|
color = gajim.config.get_per('themes', theme, 'state_muc_directed_msg')
|
|
elif chatstate:
|
|
if chatstate == 'active' or (current_tab and has_focus):
|
|
if jid in self.muc_attentions:
|
|
self.muc_attentions.remove(jid)
|
|
color = gajim.config.get_per('themes', theme, 'state_active_color')
|
|
elif chatstate == 'newmsg' and (not has_focus or not current_tab) and\
|
|
jid not in self.muc_attentions:
|
|
color = gajim.config.get_per('themes', theme, 'state_muc_msg')
|
|
if color:
|
|
color = gtk.gdk.colormap_get_system().alloc_color(color)
|
|
# The widget state depend on whether this tab is the "current" tab
|
|
if current_tab:
|
|
nickname.modify_fg(gtk.STATE_NORMAL, color)
|
|
else:
|
|
nickname.modify_fg(gtk.STATE_ACTIVE, color)
|
|
|
|
if gajim.config.get('tabs_close_button'):
|
|
close_button.show()
|
|
else:
|
|
close_button.hide()
|
|
|
|
nickname.set_max_width_chars(10)
|
|
nickname.set_text(unread + self.names[jid])
|
|
|
|
|
|
def on_window_destroy(self, widget, kind): #kind is 'chats' or 'gc'
|
|
'''clean self.plugin.windows[self.account][kind]'''
|
|
for jid in self.xmls:
|
|
windows = self.plugin.windows[self.account][kind]
|
|
if kind == 'chats':
|
|
# send 'gone' chatstate to every tabbed chat tab
|
|
windows[jid].send_chatstate('gone', jid)
|
|
gobject.source_remove(self.possible_paused_timeout_id[jid])
|
|
gobject.source_remove(self.possible_inactive_timeout_id[jid])
|
|
if self.plugin.systray_enabled and self.nb_unread[jid] > 0:
|
|
self.plugin.systray.remove_jid(jid, self.account)
|
|
del windows[jid]
|
|
if self.print_time_timeout_id.has_key(jid):
|
|
gobject.source_remove(self.print_time_timeout_id[jid])
|
|
if windows.has_key('tabbed'):
|
|
del windows['tabbed']
|
|
|
|
def get_active_jid(self):
|
|
notebook = self.notebook
|
|
active_child = notebook.get_nth_page(notebook.get_current_page())
|
|
active_jid = ''
|
|
for jid in self.xmls:
|
|
if self.childs[jid] == active_child:
|
|
active_jid = jid
|
|
break
|
|
return active_jid
|
|
|
|
def on_close_button_clicked(self, button, jid):
|
|
"""When close button is pressed: close a tab"""
|
|
self.remove_tab(jid)
|
|
|
|
def on_history_menuitem_clicked(self, widget = None, jid = None):
|
|
"""When history menuitem is pressed: call history window"""
|
|
if jid is None:
|
|
jid = self.get_active_jid()
|
|
if self.plugin.windows['logs'].has_key(jid):
|
|
self.plugin.windows['logs'][jid].window.present()
|
|
else:
|
|
self.plugin.windows['logs'][jid] = history_window.HistoryWindow(
|
|
self.plugin, jid, self.account)
|
|
|
|
def on_chat_window_focus_in_event(self, widget, event):
|
|
"""When window gets focus"""
|
|
jid = self.get_active_jid()
|
|
|
|
textview = self.xmls[jid].get_widget('conversation_textview')
|
|
buffer = textview.get_buffer()
|
|
end_iter = buffer.get_end_iter()
|
|
end_rect = textview.get_iter_location(end_iter)
|
|
visible_rect = textview.get_visible_rect()
|
|
if end_rect.y <= (visible_rect.y + visible_rect.height):
|
|
#we are at the end
|
|
if self.nb_unread[jid] > 0:
|
|
self.nb_unread[jid] = 0
|
|
self.show_title()
|
|
if self.plugin.systray_enabled:
|
|
self.plugin.systray.remove_jid(jid, self.account)
|
|
|
|
'''TC/GC 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
|
|
# Undo "unread" state display, etc.
|
|
if self.widget_name == 'groupchat_window':
|
|
self.redraw_tab(jid, 'active')
|
|
else:
|
|
# NOTE: we do not send any chatstate to preserve inactive, gone, etc.
|
|
self.redraw_tab(jid)
|
|
|
|
def on_compact_view_menuitem_activate(self, widget):
|
|
isactive = widget.get_active()
|
|
self.set_compact_view(isactive)
|
|
|
|
def on_actions_button_clicked(self, widget):
|
|
'''popup action menu'''
|
|
menu = self.prepare_context_menu()
|
|
self.popup_is_shown = True
|
|
menu.connect('deactivate', self.on_popup_deactivate)
|
|
menu.popup(None, None, None, 1, 0)
|
|
menu.show_all()
|
|
|
|
def remove_possible_switch_to_menuitems(self, menu):
|
|
''' remove duplicate 'Switch to' if they exist and return clean menu'''
|
|
childs = menu.get_children()
|
|
|
|
if self.widget_name == 'tabbed_chat_window':
|
|
jid = self.get_active_jid()
|
|
c = gajim.get_first_contact_instance_from_jid(self.account, jid)
|
|
if _('not in the roster') in c.groups: # for add_to_roster_menuitem
|
|
childs[5].show()
|
|
childs[5].set_no_show_all(False)
|
|
else:
|
|
childs[5].hide()
|
|
childs[5].set_no_show_all(True)
|
|
|
|
start_removing_from = 6 # this is from the seperator and after
|
|
|
|
else:
|
|
start_removing_from = 7 # # this is from the seperator and after
|
|
|
|
for child in childs[start_removing_from:]:
|
|
menu.remove(child)
|
|
|
|
return menu
|
|
|
|
def prepare_context_menu(self):
|
|
'''sets compact view menuitem active state
|
|
sets active and sensitivity state for toggle_gpg_menuitem
|
|
and remove possible 'Switch to' menuitems'''
|
|
if self.widget_name == 'groupchat_window':
|
|
menu = self.gc_popup_menu
|
|
childs = menu.get_children()
|
|
# compact_view_menuitem
|
|
childs[5].set_active(self.compact_view_current_state)
|
|
elif self.widget_name == 'tabbed_chat_window':
|
|
menu = self.tabbed_chat_popup_menu
|
|
childs = menu.get_children()
|
|
# check if gpg capabitlies or else make gpg toggle insensitive
|
|
jid = self.get_active_jid()
|
|
gpg_btn = self.xmls[jid].get_widget('gpg_togglebutton')
|
|
isactive = gpg_btn.get_active()
|
|
issensitive = gpg_btn.get_property('sensitive')
|
|
childs[3].set_active(isactive)
|
|
childs[3].set_property('sensitive', issensitive)
|
|
# compact_view_menuitem
|
|
childs[4].set_active(self.compact_view_current_state)
|
|
menu = self.remove_possible_switch_to_menuitems(menu)
|
|
|
|
return menu
|
|
|
|
def popup_menu(self, event):
|
|
self.popup_is_shown = True
|
|
menu = self.prepare_context_menu()
|
|
menu.connect('deactivate', self.on_popup_deactivate)
|
|
# common menuitems (tab switches)
|
|
if len(self.xmls) > 1: # if there is more than one tab
|
|
menu.append(gtk.SeparatorMenuItem()) # seperator
|
|
for jid in self.xmls:
|
|
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_banner_eventbox_button_press_event(self, widget, event):
|
|
'''If right-clicked, show popup'''
|
|
if event.button == 3: # right click
|
|
self.popup_menu(event)
|
|
|
|
def on_popup_deactivate(self, widget):
|
|
self.popup_is_shown = False
|
|
|
|
def on_chat_notebook_switch_page(self, notebook, page, page_num):
|
|
# get the index of the page and then the page that we're leaving
|
|
old_no = notebook.get_current_page()
|
|
old_child = notebook.get_nth_page(old_no)
|
|
|
|
new_child = notebook.get_nth_page(page_num)
|
|
|
|
old_jid = ''
|
|
new_jid = ''
|
|
for jid in self.xmls:
|
|
if self.childs[jid] == new_child:
|
|
new_jid = jid
|
|
elif self.childs[jid] == old_child:
|
|
old_jid = jid
|
|
|
|
if old_jid != '' and new_jid != '': # we found both jids
|
|
break # so stop looping
|
|
|
|
if self.widget_name == 'tabbed_chat_window':
|
|
# send chatstate inactive to the one we're leaving
|
|
# and active to the one we visit
|
|
if old_jid != '':
|
|
self.send_chatstate('inactive', old_jid)
|
|
self.send_chatstate('active', new_jid)
|
|
|
|
conversation_textview = self.xmls[new_jid].get_widget(
|
|
'conversation_textview')
|
|
conversation_buffer = conversation_textview.get_buffer()
|
|
end_iter = conversation_buffer.get_end_iter()
|
|
end_rect = conversation_textview.get_iter_location(end_iter)
|
|
visible_rect = conversation_textview.get_visible_rect()
|
|
if end_rect.y <= (visible_rect.y + visible_rect.height):
|
|
#we are at the end
|
|
if self.nb_unread[new_jid] > 0:
|
|
self.nb_unread[new_jid] = 0
|
|
self.redraw_tab(new_jid)
|
|
self.show_title()
|
|
if self.plugin.systray_enabled:
|
|
self.plugin.systray.remove_jid(new_jid, self.account)
|
|
|
|
conversation_textview.grab_focus()
|
|
|
|
def set_active_tab(self, jid):
|
|
self.notebook.set_current_page(self.notebook.page_num(self.childs[jid]))
|
|
|
|
def remove_tab(self, jid, kind): #kind is 'chats' or 'gc'
|
|
if len(self.xmls) == 1: # only one tab when we asked to remove
|
|
# so destroy window
|
|
|
|
# we check and possibly save positions here, because Ctrl+W, Escape
|
|
# etc.. call remove_tab so similar code in delete_event callbacks
|
|
# is not enough
|
|
if gajim.config.get('saveposition'):
|
|
if kind == 'chats':
|
|
x, y = self.window.get_position()
|
|
gajim.config.set('chat-x-position', x)
|
|
gajim.config.set('chat-y-position', y)
|
|
width, height = self.window.get_size()
|
|
gajim.config.set('chat-width', width)
|
|
gajim.config.set('chat-height', height)
|
|
elif kind == 'gc':
|
|
gajim.config.set('gc-hpaned-position', self.hpaned_position)
|
|
x, y = self.window.get_position()
|
|
gajim.config.set('gc-x-position', x)
|
|
gajim.config.set('gc-y-position', y)
|
|
width, height = self.window.get_size()
|
|
gajim.config.set('gc-width', width)
|
|
gajim.config.set('gc-height', height)
|
|
|
|
self.window.destroy()
|
|
else:
|
|
if self.nb_unread[jid] > 0:
|
|
self.nb_unread[jid] = 0
|
|
if self.plugin.systray_enabled:
|
|
self.plugin.systray.remove_jid(jid, self.account)
|
|
if self.print_time_timeout_id.has_key(jid):
|
|
gobject.source_remove(self.print_time_timeout_id[jid])
|
|
del self.print_time_timeout_id[jid]
|
|
|
|
self.notebook.remove_page(self.notebook.page_num(self.childs[jid]))
|
|
|
|
if self.plugin.windows[self.account][kind].has_key(jid):
|
|
del self.plugin.windows[self.account][kind][jid]
|
|
del self.nb_unread[jid]
|
|
del gajim.last_message_time[self.account][jid]
|
|
del self.last_time_printout[jid]
|
|
del self.xmls[jid]
|
|
del self.childs[jid]
|
|
del self.tagIn[jid]
|
|
del self.tagOut[jid]
|
|
del self.tagStatus[jid]
|
|
del self.sent_history[jid]
|
|
del self.sent_history_pos[jid]
|
|
del self.typing_new[jid]
|
|
del self.orig_msg[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)
|
|
self.show_title()
|
|
|
|
def bring_scroll_to_end(self, textview, diff_y = 0):
|
|
''' scrolls to the end of textview if end is not visible '''
|
|
buffer = textview.get_buffer()
|
|
buffer.begin_user_action()
|
|
at_the_end = False
|
|
end_iter = buffer.get_end_iter()
|
|
end_rect = textview.get_iter_location(end_iter)
|
|
visible_rect = textview.get_visible_rect()
|
|
# scroll only if expected end is not visible
|
|
if end_rect.y >= (visible_rect.y + visible_rect.height + diff_y):
|
|
gobject.idle_add(self.scroll_to_end_iter, textview)
|
|
|
|
def scroll_to_end_iter(self, textview):
|
|
buffer = textview.get_buffer()
|
|
end_iter = buffer.get_end_iter()
|
|
textview.scroll_to_iter(end_iter, 0, False, 1, 1)
|
|
return False
|
|
|
|
def size_request(self, message_textview , requisition, xml_top):
|
|
''' When message_textview changes its size. If the new height
|
|
will enlarge the window, enable the scrollbar automatic policy'''
|
|
if message_textview.window is None:
|
|
return
|
|
message_scrolledwindow = xml_top.get_widget('message_scrolledwindow')
|
|
conversation_scrolledwindow = \
|
|
xml_top.get_widget('conversation_scrolledwindow')
|
|
conversation_textview = \
|
|
xml_top.get_widget('conversation_textview')
|
|
|
|
min_height = conversation_scrolledwindow.get_property('height-request')
|
|
conversation_height = conversation_textview.window.get_size()[1]
|
|
message_height = message_textview.window.get_size()[1]
|
|
# new tab is not exposed yet
|
|
if conversation_height < 2:
|
|
return
|
|
|
|
if conversation_height < min_height:
|
|
min_height = conversation_height
|
|
|
|
diff_y = message_height - requisition.height
|
|
if diff_y is not 0:
|
|
if conversation_height + diff_y < min_height:
|
|
if message_height + conversation_height - min_height > min_height:
|
|
message_scrolledwindow.set_property('vscrollbar-policy',
|
|
gtk.POLICY_AUTOMATIC)
|
|
message_scrolledwindow.set_property('hscrollbar-policy',
|
|
gtk.POLICY_AUTOMATIC)
|
|
message_scrolledwindow.set_property('height-request',
|
|
message_height + conversation_height - min_height)
|
|
self.bring_scroll_to_end(message_textview)
|
|
else:
|
|
message_scrolledwindow.set_property('vscrollbar-policy',
|
|
gtk.POLICY_NEVER)
|
|
message_scrolledwindow.set_property('hscrollbar-policy',
|
|
gtk.POLICY_NEVER)
|
|
message_scrolledwindow.set_property('height-request', -1)
|
|
self.bring_scroll_to_end(conversation_textview, diff_y - 18)
|
|
return True
|
|
|
|
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 new_tab(self, jid):
|
|
#FIXME: text formating buttons will be hidden in 0.8 release
|
|
for w in ['bold_togglebutton', 'italic_togglebutton', 'underline_togglebutton']:
|
|
self.xmls[jid].get_widget(w).set_no_show_all(True)
|
|
|
|
self.set_compact_view(self.always_compact_view)
|
|
self.nb_unread[jid] = 0
|
|
gajim.last_message_time[self.account][jid] = 0
|
|
self.last_time_printout[jid] = 0.
|
|
font = pango.FontDescription(gajim.config.get('conversation_font'))
|
|
|
|
if gajim.config.get('use_speller') and 'gtkspell' in globals():
|
|
message_textview = self.xmls[jid].get_widget('message_textview')
|
|
try:
|
|
gtkspell.Spell(message_textview)
|
|
except gobject.GError, msg:
|
|
#FIXME: add a ui for this use spell.set_language()
|
|
dialogs.ErrorDialog(unicode(msg), _('If that is not your language for which you want to highlight misspelled words, then please set your $LANG as appropriate. Eg. for French do export LANG=fr_FR or export LANG=fr_FR.UTF-8 in ~/.bash_profile or to make it global in /etc/profile.\n\nHighlighting misspelled words feature will not be used')).get_response()
|
|
gajim.config.set('use_speller', False)
|
|
|
|
conversation_textview = self.xmls[jid].get_widget(
|
|
'conversation_textview')
|
|
conversation_textview.modify_font(font)
|
|
conversation_buffer = conversation_textview.get_buffer()
|
|
end_iter = conversation_buffer.get_end_iter()
|
|
|
|
conversation_buffer.create_mark('end', end_iter, False)
|
|
|
|
self.tagIn[jid] = conversation_buffer.create_tag('incoming')
|
|
color = gajim.config.get('inmsgcolor')
|
|
self.tagIn[jid].set_property('foreground', color)
|
|
self.tagOut[jid] = conversation_buffer.create_tag('outgoing')
|
|
color = gajim.config.get('outmsgcolor')
|
|
self.tagOut[jid].set_property('foreground', color)
|
|
self.tagStatus[jid] = conversation_buffer.create_tag('status')
|
|
color = gajim.config.get('statusmsgcolor')
|
|
self.tagStatus[jid].set_property('foreground', color)
|
|
|
|
tag = conversation_buffer.create_tag('marked')
|
|
color = gajim.config.get('markedmsgcolor')
|
|
tag.set_property('foreground', color)
|
|
tag.set_property('weight', pango.WEIGHT_BOLD)
|
|
|
|
tag = conversation_buffer.create_tag('time_sometimes')
|
|
tag.set_property('foreground', '#9e9e9e')
|
|
tag.set_property('scale', pango.SCALE_SMALL)
|
|
tag.set_property('justification', gtk.JUSTIFY_CENTER)
|
|
|
|
tag = conversation_buffer.create_tag('small')
|
|
tag.set_property('scale', pango.SCALE_SMALL)
|
|
|
|
tag = conversation_buffer.create_tag('grey')
|
|
tag.set_property('foreground', '#9e9e9e')
|
|
|
|
tag = conversation_buffer.create_tag('url')
|
|
tag.set_property('foreground', '#0000ff')
|
|
tag.set_property('underline', pango.UNDERLINE_SINGLE)
|
|
tag.connect('event', self.hyperlink_handler, 'url')
|
|
|
|
tag = conversation_buffer.create_tag('mail')
|
|
tag.set_property('foreground', '#0000ff')
|
|
tag.set_property('underline', pango.UNDERLINE_SINGLE)
|
|
tag.connect('event', self.hyperlink_handler, 'mail')
|
|
|
|
tag = conversation_buffer.create_tag('bold')
|
|
tag.set_property('weight', pango.WEIGHT_BOLD)
|
|
|
|
tag = conversation_buffer.create_tag('italic')
|
|
tag.set_property('style', pango.STYLE_ITALIC)
|
|
|
|
tag = conversation_buffer.create_tag('underline')
|
|
tag.set_property('underline', pango.UNDERLINE_SINGLE)
|
|
|
|
self.xmls[jid].signal_autoconnect(self)
|
|
conversation_scrolledwindow = self.xmls[jid].get_widget(
|
|
'conversation_scrolledwindow')
|
|
conversation_scrolledwindow.get_vadjustment().connect('value-changed',
|
|
self.on_conversation_vadjustment_value_changed)
|
|
|
|
|
|
if len(self.xmls) > 1:
|
|
self.notebook.set_show_tabs(True)
|
|
|
|
if self.widget_name == 'tabbed_chat_window':
|
|
xm = gtk.glade.XML(GTKGUI_GLADE, 'chats_eventbox', APP)
|
|
tab_hbox = xm.get_widget('chats_eventbox')
|
|
elif self.widget_name == 'groupchat_window':
|
|
xm = gtk.glade.XML(GTKGUI_GLADE, 'gc_eventbox', APP)
|
|
tab_hbox = xm.get_widget('gc_eventbox')
|
|
|
|
child = self.childs[jid]
|
|
|
|
xm.signal_connect('on_close_button_clicked',
|
|
self.on_close_button_clicked, jid)
|
|
xm.signal_connect('on_tab_eventbox_button_press_event',
|
|
self.on_tab_eventbox_button_press_event, child)
|
|
|
|
self.notebook.append_page(child, tab_hbox)
|
|
message_textview = self.xmls[jid].get_widget('message_textview')
|
|
message_textview.modify_font(font)
|
|
message_textview.connect('size-request', self.size_request,
|
|
self.xmls[jid])
|
|
#init new sent history for this conversation
|
|
self.sent_history[jid] = []
|
|
self.sent_history_pos[jid] = 0
|
|
self.typing_new[jid] = True
|
|
self.orig_msg[jid] = ''
|
|
|
|
self.show_title()
|
|
|
|
def on_conversation_textview_key_press_event(self, widget, event):
|
|
"""Do not block these events and send them to the notebook"""
|
|
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)
|
|
elif event.keyval == gtk.keysyms.l or \
|
|
event.keyval == gtk.keysyms.L: # CTRL + L
|
|
jid = self.get_active_jid()
|
|
conversation_textview = self.xmls[jid].get_widget('conversation_textview')
|
|
conversation_textview.get_buffer().set_text('')
|
|
elif event.keyval == gtk.keysyms.v: # CTRL + V
|
|
jid = self.get_active_jid()
|
|
message_textview = self.xmls[jid].get_widget('message_textview')
|
|
if not message_textview.is_focus():
|
|
message_textview.grab_focus()
|
|
message_textview.emit('key_press_event', event)
|
|
|
|
def on_chat_notebook_key_press_event(self, widget, event):
|
|
st = '1234567890' # alt+1 means the first tab (tab 0)
|
|
jid = self.get_active_jid()
|
|
if event.keyval == gtk.keysyms.Escape: # ESCAPE
|
|
if self.widget_name == 'tabbed_chat_window':
|
|
self.remove_tab(jid)
|
|
elif event.keyval == gtk.keysyms.F4 and \
|
|
(event.state & gtk.gdk.CONTROL_MASK): # CTRL + F4
|
|
self.remove_tab(jid)
|
|
elif event.keyval == gtk.keysyms.w and \
|
|
(event.state & gtk.gdk.CONTROL_MASK): # CTRL + W
|
|
self.remove_tab(jid)
|
|
elif event.string and event.string in st and \
|
|
(event.state & gtk.gdk.MOD1_MASK): # alt + 1,2,3..
|
|
self.notebook.set_current_page(st.index(event.string))
|
|
elif event.keyval == gtk.keysyms.c and \
|
|
(event.state & gtk.gdk.MOD1_MASK): # alt + C toggles compact view
|
|
self.set_compact_view(not self.compact_view_current_state)
|
|
elif event.keyval == gtk.keysyms.Page_Down:
|
|
if event.state & gtk.gdk.SHIFT_MASK: # SHIFT + PAGE DOWN
|
|
conversation_textview = self.xmls[jid].\
|
|
get_widget('conversation_textview')
|
|
rect = conversation_textview.get_visible_rect()
|
|
iter = conversation_textview.get_iter_at_location(rect.x,\
|
|
rect.y + rect.height)
|
|
conversation_textview.scroll_to_iter(iter, 0.1, True, 0, 0)
|
|
elif event.keyval == gtk.keysyms.Page_Up:
|
|
if event.state & gtk.gdk.SHIFT_MASK: # SHIFT + PAGE UP
|
|
conversation_textview = self.xmls[jid].\
|
|
get_widget('conversation_textview')
|
|
rect = conversation_textview.get_visible_rect()
|
|
iter = conversation_textview.get_iter_at_location(rect.x, rect.y)
|
|
conversation_textview.scroll_to_iter(iter, 0.1, True, 0, 1)
|
|
# or event.keyval == gtk.keysyms.KP_Up
|
|
elif event.keyval == gtk.keysyms.Up:
|
|
if event.state & gtk.gdk.SHIFT_MASK: # SHIFT + UP
|
|
conversation_scrolledwindow = self.xml.get_widget('conversation_scrolledwindow')
|
|
conversation_scrolledwindow.emit('scroll-child',
|
|
gtk.SCROLL_PAGE_BACKWARD, False)
|
|
elif event.keyval == gtk.keysyms.ISO_Left_Tab: # SHIFT + TAB
|
|
if event.state & gtk.gdk.CONTROL_MASK: # CTRL + SHIFT + TAB
|
|
current = self.notebook.get_current_page()
|
|
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)
|
|
elif event.keyval == gtk.keysyms.Tab: # TAB
|
|
if event.state & gtk.gdk.CONTROL_MASK: # CTRL + TAB
|
|
current = self.notebook.get_current_page()
|
|
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)
|
|
elif (event.keyval == gtk.keysyms.l or event.keyval == gtk.keysyms.L) \
|
|
and event.state & gtk.gdk.CONTROL_MASK: # CTRL + L
|
|
conversation_textview = self.xmls[jid].\
|
|
get_widget('conversation_textview')
|
|
conversation_textview.get_buffer().set_text('')
|
|
elif event.keyval == gtk.keysyms.v and event.state & gtk.gdk.CONTROL_MASK:
|
|
# CTRL + V
|
|
jid = self.get_active_jid()
|
|
message_textview = self.xmls[jid].get_widget('message_textview')
|
|
if not message_textview.is_focus():
|
|
message_textview.grab_focus()
|
|
message_textview.emit('key_press_event', event)
|
|
elif event.state & gtk.gdk.CONTROL_MASK or \
|
|
(event.keyval == gtk.keysyms.Control_L) or \
|
|
(event.keyval == gtk.keysyms.Control_R):
|
|
# we pressed a control key or ctrl+sth: we don't block
|
|
# the event in order to let ctrl+c (copy text) and
|
|
# others do their default work
|
|
pass
|
|
else: # it's a normal key press make sure message_textview has focus
|
|
message_textview = self.xmls[jid].get_widget('message_textview')
|
|
if not message_textview.is_focus():
|
|
message_textview.grab_focus()
|
|
message_textview.emit('key_press_event', event)
|
|
|
|
def on_conversation_vadjustment_value_changed(self, widget):
|
|
jid = self.get_active_jid()
|
|
if not self.nb_unread[jid]:
|
|
return
|
|
textview = self.xmls[jid].get_widget('conversation_textview')
|
|
buffer = textview.get_buffer()
|
|
end_iter = buffer.get_end_iter()
|
|
end_rect = textview.get_iter_location(end_iter)
|
|
visible_rect = textview.get_visible_rect()
|
|
if end_rect.y <= (visible_rect.y + visible_rect.height) and \
|
|
self.window.is_active():
|
|
#we are at the end
|
|
self.nb_unread[jid] = 0
|
|
self.redraw_tab(jid)
|
|
self.show_title()
|
|
if self.plugin.systray_enabled:
|
|
self.plugin.systray.remove_jid(jid, self.account)
|
|
|
|
def on_conversation_textview_motion_notify_event(self, widget, event):
|
|
'''change the cursor to a hand when we are over a mail or an url'''
|
|
jid = self.get_active_jid()
|
|
x, y, spam = widget.window.get_pointer()
|
|
x, y = widget.window_to_buffer_coords(gtk.TEXT_WINDOW_TEXT, x, y)
|
|
tags = widget.get_iter_at_location(x, y).get_tags()
|
|
if self.change_cursor:
|
|
widget.get_window(gtk.TEXT_WINDOW_TEXT).set_cursor(
|
|
gtk.gdk.Cursor(gtk.gdk.XTERM))
|
|
self.change_cursor = None
|
|
tag_table = widget.get_buffer().get_tag_table()
|
|
for tag in tags:
|
|
if tag == tag_table.lookup('url') or tag == tag_table.lookup('mail'):
|
|
widget.get_window(gtk.TEXT_WINDOW_TEXT).set_cursor(
|
|
gtk.gdk.Cursor(gtk.gdk.HAND2))
|
|
self.change_cursor = tag
|
|
return False
|
|
|
|
def on_clear(self, widget, textview):
|
|
'''clear text in the given textview'''
|
|
buffer = textview.get_buffer()
|
|
start, end = buffer.get_bounds()
|
|
buffer.delete(start, end)
|
|
|
|
def visit_url_from_menuitem(self, widget, link):
|
|
'''basically it filters out the widget instance'''
|
|
helpers.launch_browser_mailer('url', link)
|
|
|
|
def on_message_textview_populate_popup(self, textview, menu):
|
|
self.popup_is_shown = True
|
|
menu.connect('deactivate', self.on_popup_deactivate)
|
|
|
|
def on_conversation_textview_populate_popup(self, textview, menu):
|
|
'''we override the default context menu and we prepend Clear
|
|
and if we have sth selected we show a submenu with actions on the phrase
|
|
(see on_conversation_textview_button_press_event)'''
|
|
self.popup_is_shown = True
|
|
menu.connect('deactivate', self.on_popup_deactivate)
|
|
item = gtk.SeparatorMenuItem()
|
|
menu.prepend(item)
|
|
item = gtk.ImageMenuItem(gtk.STOCK_CLEAR)
|
|
menu.prepend(item)
|
|
item.connect('activate', self.on_clear, textview)
|
|
if self.selected_phrase:
|
|
s = self.selected_phrase
|
|
if len(s) > 25:
|
|
s = s[:21] + '...'
|
|
item = gtk.MenuItem(_('Actions for "%s"') % s)
|
|
menu.prepend(item)
|
|
submenu = gtk.Menu()
|
|
item.set_submenu(submenu)
|
|
|
|
always_use_en = gajim.config.get('always_english_wikipedia')
|
|
if always_use_en:
|
|
link = 'http://en.wikipedia.org/wiki/Special:Search?search=%s'\
|
|
% self.selected_phrase
|
|
else:
|
|
link = 'http://%s.wikipedia.org/wiki/Special:Search?search=%s'\
|
|
% (gajim.LANG, self.selected_phrase)
|
|
item = gtk.MenuItem(_('Read _Wikipedia Article'))
|
|
item.connect('activate', self.visit_url_from_menuitem, link)
|
|
submenu.append(item)
|
|
|
|
item = gtk.MenuItem(_('Look it up in _Dictionary'))
|
|
dict_link = gajim.config.get('dictionary_url')
|
|
if dict_link == 'WIKTIONARY':
|
|
# special link (yeah undocumented but default)
|
|
always_use_en = gajim.config.get('always_english_wiktionary')
|
|
if always_use_en:
|
|
link = 'http://en.wiktionary.org/wiki/Special:Search?search=%s'\
|
|
% self.selected_phrase
|
|
else:
|
|
link = 'http://%s.wiktionary.org/wiki/Special:Search?search=%s'\
|
|
% (gajim.LANG, self.selected_phrase)
|
|
item.connect('activate', self.visit_url_from_menuitem, link)
|
|
else:
|
|
if dict_link.find('%s') == -1:
|
|
#we must have %s in the url if not WIKTIONARY
|
|
item = gtk.MenuItem(_('Dictionary URL is missing an "%s" and it is not WIKTIONARY'))
|
|
item.set_property('sensitive', False)
|
|
else:
|
|
link = dict_link % self.selected_phrase
|
|
item.connect('activate', self.visit_url_from_menuitem, link)
|
|
submenu.append(item)
|
|
|
|
|
|
search_link = gajim.config.get('search_engine')
|
|
if search_link.find('%s') == -1:
|
|
#we must have %s in the url
|
|
item = gtk.MenuItem(_('Web Search URL is missing an "%s"'))
|
|
item.set_property('sensitive', False)
|
|
else:
|
|
item = gtk.MenuItem(_('Web _Search for it'))
|
|
link = search_link % self.selected_phrase
|
|
item.connect('activate', self.visit_url_from_menuitem, link)
|
|
submenu.append(item)
|
|
|
|
menu.show_all()
|
|
|
|
def on_conversation_textview_button_press_event(self, widget, event):
|
|
# If we clicked on a taged text do NOT open the standard popup menu
|
|
# if normal text check if we have sth selected
|
|
|
|
self.selected_phrase = ''
|
|
|
|
if event.button != 3: # if not right click
|
|
return False
|
|
|
|
win = widget.get_window(gtk.TEXT_WINDOW_TEXT)
|
|
x, y = widget.window_to_buffer_coords(gtk.TEXT_WINDOW_TEXT,
|
|
int(event.x), int(event.y))
|
|
iter = widget.get_iter_at_location(x, y)
|
|
tags = iter.get_tags()
|
|
|
|
|
|
if tags: # we clicked on sth special (it can be status message too)
|
|
for tag in tags:
|
|
tag_name = tag.get_property('name')
|
|
if 'url' in tag_name or 'mail' in tag_name:
|
|
return True # we block normal context menu
|
|
|
|
# we check if sth was selected and if it was we assign
|
|
# selected_phrase variable
|
|
# so on_conversation_textview_populate_popup can use it
|
|
buffer = widget.get_buffer()
|
|
return_val = buffer.get_selection_bounds()
|
|
if return_val: # if sth was selected when we right-clicked
|
|
# get the selected text
|
|
start_sel, finish_sel = return_val[0], return_val[1]
|
|
self.selected_phrase = buffer.get_text(start_sel, finish_sel).decode('utf-8')
|
|
|
|
def print_time_timeout(self, jid):
|
|
if not jid in self.xmls.keys():
|
|
return False
|
|
if gajim.config.get('print_time') == 'sometimes':
|
|
textview = self.xmls[jid].get_widget('conversation_textview')
|
|
buffer = textview.get_buffer()
|
|
end_iter = buffer.get_end_iter()
|
|
tim = time.localtime()
|
|
tim_format = time.strftime('%H:%M', tim)
|
|
buffer.insert_with_tags_by_name(end_iter,
|
|
'\n' + tim_format,
|
|
'time_sometimes')
|
|
#scroll to the end of the textview
|
|
end_rect = textview.get_iter_location(end_iter)
|
|
visible_rect = textview.get_visible_rect()
|
|
if end_rect.y <= (visible_rect.y + visible_rect.height):
|
|
#we are at the end
|
|
self.scroll_to_end(textview)
|
|
return True # loop again
|
|
if self.print_time_timeout_id.has_key(jid):
|
|
del self.print_time_timeout_id[jid]
|
|
return False
|
|
|
|
def on_open_link_activate(self, widget, kind, text):
|
|
helpers.launch_browser_mailer(kind, text)
|
|
|
|
def on_copy_link_activate(self, widget, text):
|
|
clip = gtk.clipboard_get()
|
|
clip.set_text(text)
|
|
|
|
def on_start_chat_activate(self, widget, jid):
|
|
self.plugin.roster.new_chat_from_jid(self.account, jid)
|
|
|
|
def on_join_group_chat_menuitem_activate(self, widget, jid):
|
|
room, server = jid.split('@')
|
|
if self.plugin.windows[self.account].has_key('join_gc'):
|
|
instance = self.plugin.windows[self.account]['join_gc']
|
|
instance.xml.get_widget('server_entry').set_text(server)
|
|
instance.xml.get_widget('room_entry').set_text(room)
|
|
self.plugin.windows[self.account]['join_gc'].window.present()
|
|
else:
|
|
try:
|
|
self.plugin.windows[self.account]['join_gc'] = \
|
|
dialogs.JoinGroupchatWindow(self.plugin, self.account, server, room)
|
|
except RuntimeError:
|
|
pass
|
|
|
|
def on_add_to_roster_activate(self, widget, jid):
|
|
dialogs.AddNewContactWindow(self.plugin, self.account, jid)
|
|
|
|
def make_link_menu(self, event, kind, text):
|
|
xml = gtk.glade.XML(GTKGUI_GLADE, 'chat_context_menu', APP)
|
|
menu = xml.get_widget('chat_context_menu')
|
|
self.popup_is_shown = True
|
|
menu.connect('deactivate', self.on_popup_deactivate)
|
|
childs = menu.get_children()
|
|
if kind == 'url':
|
|
childs[0].connect('activate', self.on_copy_link_activate, text)
|
|
childs[1].connect('activate', self.on_open_link_activate, kind, text)
|
|
childs[2].hide() # copy mail address
|
|
childs[3].hide() # open mail composer
|
|
childs[4].hide() # jid section seperator
|
|
childs[5].hide() # start chat
|
|
childs[6].hide() # join group chat
|
|
childs[7].hide() # add to roster
|
|
else: # It's a mail or a JID
|
|
childs[2].connect('activate', self.on_copy_link_activate, text)
|
|
childs[3].connect('activate', self.on_open_link_activate, kind, text)
|
|
childs[5].connect('activate', self.on_start_chat_activate, text)
|
|
childs[6].connect('activate',
|
|
self.on_join_group_chat_menuitem_activate, text)
|
|
|
|
allow_add = False
|
|
if gajim.contacts[self.account].has_key(text):
|
|
c = gajim.contacts[self.account][text][0]
|
|
if _('not in the roster') in c.groups:
|
|
allow_add = True
|
|
else: # he's not at all in the account contacts
|
|
allow_add = True
|
|
|
|
if allow_add:
|
|
childs[7].connect('activate', self.on_add_to_roster_activate, text)
|
|
childs[7].show() # show add to roster menuitem
|
|
else:
|
|
childs[7].hide() # hide add to roster menuitem
|
|
|
|
childs[0].hide() # copy link location
|
|
childs[1].hide() # open link in browser
|
|
|
|
menu.popup(None, None, None, event.button, event.time)
|
|
|
|
def hyperlink_handler(self, texttag, widget, event, iter, kind):
|
|
if event.type == gtk.gdk.BUTTON_PRESS:
|
|
begin_iter = iter.copy()
|
|
#we get the begining of the tag
|
|
while not begin_iter.begins_tag(texttag):
|
|
begin_iter.backward_char()
|
|
end_iter = iter.copy()
|
|
#we get the end of the tag
|
|
while not end_iter.ends_tag(texttag):
|
|
end_iter.forward_char()
|
|
word = widget.get_buffer().get_text(begin_iter, end_iter).decode(
|
|
'utf-8')
|
|
if event.button == 3: # right click
|
|
self.make_link_menu(event, kind, word)
|
|
else:
|
|
#we launch the correct application
|
|
helpers.launch_browser_mailer(kind, word)
|
|
|
|
def detect_and_print_special_text(self, otext, jid, other_tags):
|
|
textview = self.xmls[jid].get_widget('conversation_textview')
|
|
buffer = textview.get_buffer()
|
|
|
|
start = 0
|
|
end = 0
|
|
index = 0
|
|
|
|
# basic: links + mail + formatting is always checked (we like that)
|
|
if gajim.config.get('useemoticons'): # search for emoticons & urls
|
|
iterator = self.plugin.emot_and_basic_re.finditer(otext)
|
|
else: # search for just urls + mail + formatting
|
|
iterator = self.plugin.basic_pattern_re.finditer(otext)
|
|
for match in iterator:
|
|
start, end = match.span()
|
|
special_text = otext[start:end]
|
|
if start != 0:
|
|
text_before_special_text = otext[index:start]
|
|
end_iter = buffer.get_end_iter()
|
|
buffer.insert_with_tags_by_name(end_iter,
|
|
text_before_special_text, *other_tags)
|
|
index = end # update index
|
|
|
|
# now print it
|
|
self.print_special_text(special_text, other_tags, textview)
|
|
|
|
return index
|
|
|
|
def print_special_text(self, special_text, other_tags, textview):
|
|
tags = []
|
|
use_other_tags = True
|
|
show_ascii_formatting_chars=gajim.config.get('show_ascii_formatting_chars')
|
|
buffer = textview.get_buffer()
|
|
|
|
possible_emot_ascii_caps = special_text.upper() # emoticons keys are CAPS
|
|
if possible_emot_ascii_caps in self.plugin.emoticons.keys():
|
|
#it's an emoticon
|
|
emot_ascii = possible_emot_ascii_caps
|
|
end_iter = buffer.get_end_iter()
|
|
anchor = buffer.create_child_anchor(end_iter)
|
|
img = gtk.Image()
|
|
img.set_from_file(self.plugin.emoticons[emot_ascii])
|
|
img.show()
|
|
#add with possible animation
|
|
textview.add_child_at_anchor(img, anchor)
|
|
elif special_text.startswith('mailto:'):
|
|
#it's a mail
|
|
tags.append('mail')
|
|
use_other_tags = False
|
|
elif self.plugin.sth_at_sth_dot_sth_re.match(special_text):
|
|
#it's a mail
|
|
tags.append('mail')
|
|
use_other_tags = False
|
|
elif special_text.startswith('*'): # it's a bold text
|
|
tags.append('bold')
|
|
if special_text[1] == '/': # it's also italic
|
|
tags.append('italic')
|
|
if not show_ascii_formatting_chars:
|
|
special_text = special_text[2:-2] # remove */ /*
|
|
elif special_text[1] == '_': # it's also underlined
|
|
tags.append('underline')
|
|
if not show_ascii_formatting_chars:
|
|
special_text = special_text[2:-2] # remove *_ _*
|
|
else:
|
|
if not show_ascii_formatting_chars:
|
|
special_text = special_text[1:-1] # remove * *
|
|
elif special_text.startswith('/'): # it's an italic text
|
|
tags.append('italic')
|
|
if special_text[1] == '*': # it's also bold
|
|
tags.append('bold')
|
|
if not show_ascii_formatting_chars:
|
|
special_text = special_text[2:-2] # remove /* */
|
|
elif special_text[1] == '_': # it's also underlined
|
|
tags.append('underline')
|
|
if not show_ascii_formatting_chars:
|
|
special_text = special_text[2:-2] # remove /_ _/
|
|
else:
|
|
if not show_ascii_formatting_chars:
|
|
special_text = special_text[1:-1] # remove / /
|
|
elif special_text.startswith('_'): # it's an underlined text
|
|
tags.append('underline')
|
|
if special_text[1] == '*': # it's also bold
|
|
tags.append('bold')
|
|
if not show_ascii_formatting_chars:
|
|
special_text = special_text[2:-2] # remove _* *_
|
|
elif special_text[1] == '/': # it's also italic
|
|
tags.append('italic')
|
|
if not show_ascii_formatting_chars:
|
|
special_text = special_text[2:-2] # remove _/ /_
|
|
else:
|
|
if not show_ascii_formatting_chars:
|
|
special_text = special_text[1:-1] # remove _ _
|
|
else:
|
|
#it's a url
|
|
tags.append('url')
|
|
use_other_tags = False
|
|
|
|
if len(tags) > 0:
|
|
end_iter = buffer.get_end_iter()
|
|
all_tags = tags[:]
|
|
if use_other_tags:
|
|
all_tags += other_tags
|
|
buffer.insert_with_tags_by_name(end_iter, special_text, *all_tags)
|
|
|
|
def scroll_to_end(self, textview):
|
|
parent = textview.get_parent()
|
|
buffer = textview.get_buffer()
|
|
textview.scroll_to_mark(buffer.get_mark('end'), 0, True, 0, 1)
|
|
adjustment = parent.get_hadjustment()
|
|
adjustment.set_value(0)
|
|
return False
|
|
|
|
def print_empty_line(self, jid):
|
|
textview = self.xmls[jid].get_widget('conversation_textview')
|
|
buffer = textview.get_buffer()
|
|
end_iter = buffer.get_end_iter()
|
|
buffer.insert(end_iter, '\n')
|
|
|
|
def print_conversation_line(self, text, jid, kind, name, tim,
|
|
other_tags_for_name = [], other_tags_for_time = [],
|
|
other_tags_for_text = [], count_as_new = True, subject = None):
|
|
'''' prints 'chat' type messages '''
|
|
textview = self.xmls[jid].get_widget('conversation_textview')
|
|
buffer = textview.get_buffer()
|
|
buffer.begin_user_action()
|
|
at_the_end = False
|
|
end_iter = buffer.get_end_iter()
|
|
end_rect = textview.get_iter_location(end_iter)
|
|
visible_rect = textview.get_visible_rect()
|
|
if end_rect.y <= (visible_rect.y + visible_rect.height):
|
|
at_the_end = True
|
|
|
|
# FIXME: who gives us text that is not a string?
|
|
if not text:
|
|
# FIXME: Let's find out...
|
|
assert(False)
|
|
text = ''
|
|
|
|
if buffer.get_char_count() > 0:
|
|
buffer.insert(end_iter, '\n')
|
|
update_time = True
|
|
if kind == 'incoming_queue':
|
|
kind = 'incoming'
|
|
update_time = False
|
|
# print the time stamp
|
|
if gajim.config.get('print_time') == 'always':
|
|
if not tim:
|
|
tim = time.localtime()
|
|
before_str = gajim.config.get('before_time')
|
|
after_str = gajim.config.get('after_time')
|
|
format = before_str + '%H:%M:%S' + after_str
|
|
tim_format = time.strftime(format, tim)
|
|
buffer.insert_with_tags_by_name(end_iter, tim_format + ' ',
|
|
*other_tags_for_time)
|
|
elif gajim.config.get('print_time') == 'sometimes':
|
|
every_foo_seconds = 60 * gajim.config.get(
|
|
'print_ichat_every_foo_minutes')
|
|
seconds_passed = time.time() - self.last_time_printout[jid]
|
|
if seconds_passed > every_foo_seconds:
|
|
self.last_time_printout[jid] = time.time()
|
|
end_iter = buffer.get_end_iter()
|
|
tim = time.localtime()
|
|
tim_format = time.strftime('%H:%M', tim)
|
|
buffer.insert_with_tags_by_name(end_iter,
|
|
tim_format + '\n',
|
|
'time_sometimes')
|
|
#scroll to the end of the textview
|
|
end_rect = textview.get_iter_location(end_iter)
|
|
visible_rect = textview.get_visible_rect()
|
|
|
|
text_tags = other_tags_for_text[:]
|
|
if kind == 'status':
|
|
text_tags.append(kind)
|
|
elif text.startswith('/me ') or text.startswith('/me\n'):
|
|
text = '* ' + name + text[3:]
|
|
text_tags.append(kind)
|
|
|
|
if name and len(text_tags) == len(other_tags_for_text):
|
|
# not status nor /me
|
|
name_tags = other_tags_for_name[:] #create a new list
|
|
name_tags.append(kind)
|
|
before_str = gajim.config.get('before_nickname')
|
|
after_str = gajim.config.get('after_nickname')
|
|
format = before_str + name + after_str + ' '
|
|
buffer.insert_with_tags_by_name(end_iter, format, *name_tags)
|
|
|
|
# detect urls formatting and if the user has it on emoticons
|
|
index = self.detect_and_print_special_text(text, jid, text_tags)
|
|
|
|
if subject: # if we have subject, show it too!
|
|
subject = _('Subject: %s\n') % subject
|
|
end_iter = buffer.get_end_iter()
|
|
buffer.insert(end_iter, subject)
|
|
|
|
# add the rest of text located in the index and after
|
|
end_iter = buffer.get_end_iter()
|
|
buffer.insert_with_tags_by_name(end_iter, text[index:], *text_tags)
|
|
|
|
#scroll to the end of the textview
|
|
end = False
|
|
if at_the_end or kind == 'outgoing':
|
|
#we are at the end or we are sending something
|
|
end = True
|
|
# We scroll to the end after the scrollbar has appeared
|
|
gobject.idle_add(self.scroll_to_end, textview)
|
|
|
|
buffer.end_user_action()
|
|
|
|
if not count_as_new:
|
|
return
|
|
if kind == 'incoming' and update_time:
|
|
gajim.last_message_time[self.account][jid] = time.time()
|
|
urgent = True
|
|
if (jid != self.get_active_jid() or \
|
|
not self.window.is_active() or \
|
|
not end) and kind == 'incoming':
|
|
if self.widget_name == 'groupchat_window':
|
|
if text.find(self.nicks[jid]) == -1:
|
|
# Do not notify us for gc messages that are not for us
|
|
urgent = False
|
|
if not gajim.config.get('notify_on_all_muc_messages'):
|
|
return
|
|
self.nb_unread[jid] += 1
|
|
if self.plugin.systray_enabled and gajim.config.get(
|
|
'trayicon_notification_on_new_messages'):
|
|
self.plugin.systray.add_jid(jid, self.account)
|
|
self.redraw_tab(jid)
|
|
self.show_title(urgent)
|
|
|
|
def save_sent_message(self, jid, message):
|
|
#save the message, so user can scroll though the list with key up/down
|
|
size = len(self.sent_history[jid])
|
|
#we don't want size of the buffer to grow indefinately
|
|
max_size = gajim.config.get('key_up_lines')
|
|
if size >= max_size:
|
|
for i in range(0, size - 1):
|
|
self.sent_history[jid][i] = self.sent_history[jid][i+1]
|
|
self.sent_history[jid][max_size - 1] = message
|
|
else:
|
|
self.sent_history[jid].append(message)
|
|
self.sent_history_pos[jid] = size + 1
|
|
|
|
self.typing_new[jid] = True
|
|
self.orig_msg[jid] = ''
|
|
|
|
def sent_messages_scroll(self, jid, direction, conv_buf):
|
|
size = len(self.sent_history[jid])
|
|
if self.typing_new[jid]:
|
|
#user was typing something and then went into history, so save
|
|
#whatever is already typed
|
|
start_iter = conv_buf.get_start_iter()
|
|
end_iter = conv_buf.get_end_iter()
|
|
self.orig_msg[jid] = conv_buf.get_text(start_iter, end_iter, 0).decode('utf-8')
|
|
self.typing_new[jid] = False
|
|
if direction == 'up':
|
|
if self.sent_history_pos[jid] == 0:
|
|
return
|
|
self.sent_history_pos[jid] = self.sent_history_pos[jid] - 1
|
|
conv_buf.set_text(self.sent_history[jid][self.sent_history_pos[jid]])
|
|
|
|
elif direction == 'down':
|
|
if self.sent_history_pos[jid] >= size - 1:
|
|
conv_buf.set_text(self.orig_msg[jid]);
|
|
self.typing_new[jid] = True
|
|
self.sent_history_pos[jid] = size
|
|
return
|
|
|
|
self.sent_history_pos[jid] = self.sent_history_pos[jid] + 1
|
|
conv_buf.set_text(self.sent_history[jid][self.sent_history_pos[jid]])
|
|
|
|
def paint_banner(self, jid):
|
|
theme = gajim.config.get('roster_theme')
|
|
bgcolor = gajim.config.get_per('themes', theme, 'bannerbgcolor')
|
|
textcolor = gajim.config.get_per('themes', theme, 'bannertextcolor')
|
|
# the backgrounds are colored by using an eventbox by
|
|
# setting the bg color of the eventbox and the fg of the name_label
|
|
if bgcolor:
|
|
self.xmls[jid].get_widget('banner_eventbox').modify_bg(
|
|
gtk.STATE_NORMAL, gtk.gdk.color_parse(bgcolor))
|
|
if textcolor:
|
|
banner_name_label = self.xmls[jid].get_widget('banner_name_label')
|
|
banner_name_label.modify_fg(gtk.STATE_NORMAL,
|
|
gtk.gdk.color_parse(textcolor))
|
|
|
|
def repaint_colored_widgets(self):
|
|
"""Repaint widgets (banner) in the window/tab with theme color"""
|
|
# iterate through tabs/windows and repaint
|
|
for jid in self.xmls:
|
|
self.paint_banner(jid)
|
|
|
|
def set_compact_view(self, state):
|
|
'''Toggle compact view. state is bool'''
|
|
self.compact_view_current_state = state
|
|
|
|
for jid in self.xmls:
|
|
if self.widget_name == 'tabbed_chat_window':
|
|
widgets = [
|
|
self.xmls[jid].get_widget('banner_eventbox'),
|
|
self.xmls[jid].get_widget('actions_hbox'),
|
|
]
|
|
elif self.widget_name == 'groupchat_window':
|
|
widgets = [self.xmls[jid].get_widget('banner_eventbox'),
|
|
self.xmls[jid].get_widget('gc_actions_hbox'),
|
|
self.xmls[jid].get_widget('list_scrolledwindow'),
|
|
]
|
|
|
|
for widget in widgets:
|
|
if state:
|
|
widget.set_no_show_all(True)
|
|
widget.hide()
|
|
else:
|
|
widget.set_no_show_all(False)
|
|
widget.show_all()
|
|
# make the last message visible, when changing to "full view"
|
|
if not state:
|
|
conversation_textview = \
|
|
self.xmls[jid].get_widget('conversation_textview')
|
|
gobject.idle_add(self.scroll_to_end_iter, conversation_textview)
|