gajim-plural/src/tooltips.py

778 lines
25 KiB
Python
Raw Normal View History

2007-04-01 11:02:04 +02:00
# -*- coding: utf-8 -*-
## src/tooltips.py
##
## Copyright (C) 2005 Alex Mauer <hawke AT hawkesnest.net>
## Stéphan Kochen <stephan AT kochen.nl>
## Copyright (C) 2005-2006 Dimitur Kirov <dkirov AT gmail.com>
## Copyright (C) 2005-2007 Nikos Kouremenos <kourem AT gmail.com>
## Copyright (C) 2005-2008 Yann Leboulanger <asterix AT lagaule.org>
## Copyright (C) 2006 Travis Shirk <travis AT pobox.com>
## Stefan Bethge <stefan AT lanpartei.de>
## Copyright (C) 2006-2007 Jean-Marie Traissard <jim AT lapin.org>
## Copyright (C) 2007 Julien Pivotto <roidelapluie AT gmail.com>
## Copyright (C) 2007-2008 Stephan Erb <steve-e AT h3c.de>
## Copyright (C) 2008 Jonathan Schleifer <js-gajim AT webkeks.org>
##
2007-12-12 09:44:46 +01:00
## This file is part of Gajim.
##
## Gajim is free software; you can redistribute it and/or modify
## it under the terms of the GNU General Public License as published
2007-12-12 09:44:46 +01:00
## by the Free Software Foundation; version 3 only.
##
2007-12-12 09:44:46 +01:00
## Gajim 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.
##
2007-12-12 09:44:46 +01:00
## You should have received a copy of the GNU General Public License
## along with Gajim. If not, see <http://www.gnu.org/licenses/>.
2007-12-12 09:44:46 +01:00
##
import gtk
import gobject
import os
import time
import locale
import gtkgui_helpers
from common import gajim
from common import helpers
2008-07-29 21:02:57 +02:00
from common.pep import MOODS, ACTIVITIES
class BaseTooltip:
2009-11-25 21:59:43 +01:00
"""
Base Tooltip class
Usage:
tooltip = BaseTooltip()
2008-12-03 22:56:12 +01:00
....
tooltip.show_tooltip(data, widget_height, widget_y_position)
....
if tooltip.timeout != 0:
tooltip.hide_tooltip()
2008-12-03 22:56:12 +01:00
* data - the text to be displayed (extenders override this argument and
2006-10-05 02:06:57 +02:00
display more complex contents)
* widget_height - the height of the widget on which we want to show tooltip
* widget_y_position - the vertical position of the widget on the screen
2008-12-03 22:56:12 +01:00
Tooltip is displayed aligned centered to the mouse poiner and 4px below the widget.
In case tooltip goes below the visible area it is shown above the widget.
2009-11-25 21:59:43 +01:00
"""
def __init__(self):
self.timeout = 0
self.preferred_position = [0, 0]
self.win = None
self.id = None
self.cur_data = None
self.check_last_time = None
2008-12-03 22:56:12 +01:00
def populate(self, data):
2009-11-25 21:59:43 +01:00
"""
This method must be overriden by all extenders. This is the most simple
implementation: show data as value of a label
"""
self.create_window()
self.win.add(gtk.Label(data))
2008-12-03 22:56:12 +01:00
def create_window(self):
2009-11-25 21:59:43 +01:00
"""
Create a popup window each time tooltip is requested
"""
self.win = gtk.Window(gtk.WINDOW_POPUP)
self.win.set_border_width(3)
self.win.set_resizable(False)
self.win.set_name('gtk-tooltips')
self.win.set_type_hint(gtk.gdk.WINDOW_TYPE_HINT_TOOLTIP)
2008-12-03 22:56:12 +01:00
self.win.set_events(gtk.gdk.POINTER_MOTION_MASK)
self.win.connect_after('expose_event', self.expose)
self.win.connect('size-request', self.on_size_request)
self.win.connect('motion-notify-event', self.motion_notify_event)
self.screen = self.win.get_screen()
2008-12-03 22:56:12 +01:00
def _get_icon_name_for_tooltip(self, contact):
2009-11-25 21:59:43 +01:00
"""
Helper function used for tooltip contacts/acounts
Tooltip on account has fake contact with sub == '', in this case we show
real status of the account
2009-11-25 21:59:43 +01:00
"""
if contact.ask == 'subscribe':
return 'requested'
elif contact.sub in ('both', 'to', ''):
return contact.show
return 'not in roster'
def motion_notify_event(self, widget, event):
self.hide_tooltip()
def on_size_request(self, widget, requisition):
2006-02-28 11:15:42 +01:00
half_width = requisition.width / 2 + 1
2008-12-03 22:56:12 +01:00
if self.preferred_position[0] < half_width:
self.preferred_position[0] = 0
2007-12-12 09:44:46 +01:00
elif self.preferred_position[0] + requisition.width > \
self.screen.get_width() + half_width:
self.preferred_position[0] = self.screen.get_width() - \
requisition.width
elif not self.check_last_time:
2008-12-03 22:56:12 +01:00
self.preferred_position[0] -= half_width
if self.preferred_position[1] + requisition.height > \
self.screen.get_height():
# flip tooltip up
2007-12-12 09:44:46 +01:00
self.preferred_position[1] -= requisition.height + \
self.widget_height + 8
if self.preferred_position[1] < 0:
self.preferred_position[1] = 0
self.win.move(self.preferred_position[0], self.preferred_position[1])
def expose(self, widget, event):
style = self.win.get_style()
size = self.win.get_size()
2006-02-05 17:22:25 +01:00
style.paint_flat_box(self.win.window, gtk.STATE_NORMAL, gtk.SHADOW_OUT,
None, self.win, 'tooltip', 0, 0, -1, 1)
style.paint_flat_box(self.win.window, gtk.STATE_NORMAL, gtk.SHADOW_OUT,
None, self.win, 'tooltip', 0, size[1] - 1, -1, 1)
style.paint_flat_box(self.win.window, gtk.STATE_NORMAL, gtk.SHADOW_OUT,
None, self.win, 'tooltip', 0, 0, 1, -1)
style.paint_flat_box(self.win.window, gtk.STATE_NORMAL, gtk.SHADOW_OUT,
None, self.win, 'tooltip', size[0] - 1, 0, 1, -1)
return True
2008-12-03 22:56:12 +01:00
def show_tooltip(self, data, widget_height, widget_y_position):
2009-11-25 21:59:43 +01:00
"""
Show tooltip on widget
Data contains needed data for tooltip contents.
widget_height is the height of the widget on which we show the tooltip.
widget_y_position is vertical position of the widget on the screen.
"""
self.cur_data = data
2008-12-03 22:56:12 +01:00
# set tooltip contents
self.populate(data)
2008-12-03 22:56:12 +01:00
# get the X position of mouse pointer on the screen
pointer_x = self.screen.get_display().get_pointer()[1]
2008-12-03 22:56:12 +01:00
# get the prefered X position of the tooltip on the screen in case this position is >
# than the height of the screen, tooltip will be shown above the widget
preferred_y = widget_y_position + widget_height + 4
2008-12-03 22:56:12 +01:00
self.preferred_position = [pointer_x, preferred_y]
2006-10-05 02:06:57 +02:00
self.widget_height = widget_height
self.win.ensure_style()
self.win.show_all()
def hide_tooltip(self):
if self.timeout > 0:
gobject.source_remove(self.timeout)
self.timeout = 0
if self.win:
self.win.destroy()
self.win = None
self.id = None
self.cur_data = None
self.check_last_time = None
class StatusTable:
2009-11-25 21:59:43 +01:00
"""
Contains methods for creating status table. This is used in Roster and
NotificationArea tooltips
"""
def __init__(self):
self.current_row = 1
self.table = None
self.text_label = None
2005-11-09 22:05:55 +01:00
self.spacer_label = ' '
2008-12-03 22:56:12 +01:00
def create_table(self):
self.table = gtk.Table(4, 1)
2005-08-26 18:31:22 +02:00
self.table.set_property('column-spacing', 2)
2008-12-03 22:56:12 +01:00
2007-08-09 17:39:18 +02:00
def add_text_row(self, text, col_inc = 0):
self.current_row += 1
self.text_label = gtk.Label()
self.text_label.set_line_wrap(True)
self.text_label.set_alignment(0, 0)
self.text_label.set_selectable(False)
2006-03-21 16:44:36 +01:00
self.text_label.set_markup(text)
2007-08-09 17:39:18 +02:00
self.table.attach(self.text_label, 1 + col_inc, 4, self.current_row,
self.current_row + 1)
2008-12-03 22:56:12 +01:00
def get_status_info(self, resource, priority, show, status):
str_status = resource + ' (' + unicode(priority) + ')'
if status:
status = status.strip()
if status != '':
# make sure 'status' is unicode before we send to to reduce_chars
2005-09-11 15:56:38 +02:00
if isinstance(status, str):
status = unicode(status, encoding='utf-8')
# reduce to 100 chars, 1 line
status = helpers.reduce_chars_newlines(status, 100, 1)
str_status = gobject.markup_escape_text(str_status)
status = gobject.markup_escape_text(status)
str_status += ' - <i>' + status + '</i>'
return str_status
2008-12-03 22:56:12 +01:00
def add_status_row(self, file_path, show, str_status, status_time=None,
2009-11-25 21:59:43 +01:00
show_lock=False, indent=True):
"""
Append a new row with status icon to the table
"""
self.current_row += 1
state_file = show.replace(' ', '_')
files = []
files.append(os.path.join(file_path, state_file + '.png'))
files.append(os.path.join(file_path, state_file + '.gif'))
image = gtk.Image()
image.set_from_pixbuf(None)
2008-12-03 18:16:04 +01:00
for f in files:
if os.path.exists(f):
image.set_from_file(f)
break
2005-11-09 22:05:55 +01:00
spacer = gtk.Label(self.spacer_label)
image.set_alignment(1, 0.5)
if indent:
2008-12-03 22:56:12 +01:00
self.table.attach(spacer, 1, 2, self.current_row,
self.current_row + 1, 0, 0, 0, 0)
2008-12-03 22:56:12 +01:00
self.table.attach(image, 2, 3, self.current_row,
self.current_row + 1, gtk.FILL, gtk.FILL, 2, 0)
status_label = gtk.Label()
status_label.set_markup(str_status)
status_label.set_alignment(0, 0)
status_label.set_line_wrap(True)
self.table.attach(status_label, 3, 4, self.current_row,
2006-03-21 16:44:36 +01:00
self.current_row + 1, gtk.FILL | gtk.EXPAND, 0, 0, 0)
if show_lock:
lock_image = gtk.Image()
2008-12-03 22:56:12 +01:00
lock_image.set_from_stock(gtk.STOCK_DIALOG_AUTHENTICATION,
gtk.ICON_SIZE_MENU)
self.table.attach(lock_image, 4, 5, self.current_row,
self.current_row + 1, 0, 0, 0, 0)
2008-12-03 22:56:12 +01:00
class NotificationAreaTooltip(BaseTooltip, StatusTable):
2009-11-25 21:59:43 +01:00
"""
Tooltip that is shown in the notification area
"""
def __init__(self):
BaseTooltip.__init__(self)
StatusTable.__init__(self)
2005-11-09 22:05:55 +01:00
def fill_table_with_accounts(self, accounts):
iconset = gajim.config.get('iconset')
if not iconset:
2005-12-12 16:14:58 +01:00
iconset = 'dcraven'
2007-08-09 17:39:18 +02:00
file_path = os.path.join(helpers.get_iconset_path(iconset), '16x16')
2005-11-09 22:05:55 +01:00
for acct in accounts:
message = acct['message']
2008-12-03 22:56:12 +01:00
# before reducing the chars we should assure we send unicode, else
2005-11-09 22:05:55 +01:00
# there are possible pango TBs on 'set_markup'
if isinstance(message, str):
message = unicode(message, encoding = 'utf-8')
message = helpers.reduce_chars_newlines(message, 100, 1)
message = gobject.markup_escape_text(message)
if acct['name'] in gajim.con_types and \
gajim.con_types[acct['name']] in ('tls', 'ssl'):
show_lock = True
else:
show_lock = False
2005-11-09 22:05:55 +01:00
if message:
2008-12-03 22:56:12 +01:00
self.add_status_row(file_path, acct['show'],
gobject.markup_escape_text(acct['name']) + \
' - ' + message, show_lock=show_lock, indent=False)
2005-11-09 22:05:55 +01:00
else:
2008-12-03 22:56:12 +01:00
self.add_status_row(file_path, acct['show'],
gobject.markup_escape_text(acct['name'])
, show_lock=show_lock, indent=False)
2007-08-09 17:39:18 +02:00
for line in acct['event_lines']:
self.add_text_row(' ' + line, 1)
2005-11-09 22:05:55 +01:00
def populate(self, data=''):
2005-11-09 22:05:55 +01:00
self.create_window()
self.create_table()
2007-08-09 17:39:18 +02:00
accounts = helpers.get_notification_icon_tooltip_dict()
self.table.resize(2, 1)
self.fill_table_with_accounts(accounts)
2006-10-05 02:06:57 +02:00
self.hbox = gtk.HBox()
self.table.set_property('column-spacing', 1)
2005-11-09 22:05:55 +01:00
self.hbox.add(self.table)
self.hbox.show_all()
2005-08-24 19:29:35 +02:00
class GCTooltip(BaseTooltip):
2009-11-25 21:59:43 +01:00
"""
Tooltip that is shown in the GC treeview
"""
def __init__(self):
2005-08-24 14:58:03 +02:00
self.account = None
self.text_label = gtk.Label()
self.text_label.set_line_wrap(True)
self.text_label.set_alignment(0, 0)
self.text_label.set_selectable(False)
2006-03-17 16:26:58 +01:00
self.avatar_image = gtk.Image()
2005-08-24 14:58:03 +02:00
BaseTooltip.__init__(self)
2008-12-03 22:56:12 +01:00
2005-08-24 14:58:03 +02:00
def populate(self, contact):
2005-08-24 19:29:35 +02:00
if not contact:
2005-08-24 14:58:03 +02:00
return
self.create_window()
2006-03-21 22:33:56 +01:00
vcard_table = gtk.Table(3, 1)
vcard_table.set_property('column-spacing', 2)
vcard_table.set_homogeneous(False)
vcard_current_row = 1
properties = []
nick_markup = '<b>' + \
gobject.markup_escape_text(contact.get_shown_name()) \
2008-12-03 22:56:12 +01:00
+ '</b>'
properties.append((nick_markup, None))
2008-12-03 22:56:12 +01:00
if contact.status: # status message
status = contact.status.strip()
if status != '':
2005-08-24 16:08:35 +02:00
# escape markup entities
2007-08-09 17:39:18 +02:00
status = helpers.reduce_chars_newlines(status, 300, 5)
status = '<i>' +\
gobject.markup_escape_text(status) + '</i>'
properties.append((status, None))
else: # no status message, show SHOW instead
show = helpers.get_uf_show(contact.show)
show = '<i>' + show + '</i>'
properties.append((show, None))
if contact.jid.strip() != '':
properties.append((_('Jabber ID: '), contact.jid))
if hasattr(contact, 'resource') and contact.resource.strip() != '':
2008-12-03 22:56:12 +01:00
properties.append((_('Resource: '),
gobject.markup_escape_text(contact.resource) ))
if contact.affiliation != 'none':
uf_affiliation = helpers.get_uf_affiliation(contact.affiliation)
affiliation_str = \
_('%(owner_or_admin_or_member)s of this group chat') %\
{'owner_or_admin_or_member': uf_affiliation}
properties.append((affiliation_str, None))
2008-12-03 22:56:12 +01:00
2006-03-17 16:26:58 +01:00
# Add avatar
puny_name = helpers.sanitize_filename(contact.name)
puny_room = helpers.sanitize_filename(contact.room_jid)
2008-12-03 18:16:04 +01:00
file_ = helpers.get_avatar_path(os.path.join(gajim.AVATAR_PATH, puny_room,
2007-06-03 12:04:20 +02:00
puny_name))
2008-12-03 18:16:04 +01:00
if file_:
2008-12-03 21:45:26 +01:00
self.avatar_image.set_from_file(file_)
2007-06-03 12:04:20 +02:00
pix = self.avatar_image.get_pixbuf()
pix = gtkgui_helpers.get_scaled_pixbuf(pix, 'tooltip')
self.avatar_image.set_from_pixbuf(pix)
2006-03-17 16:26:58 +01:00
else:
self.avatar_image.set_from_pixbuf(None)
2006-03-21 22:33:56 +01:00
while properties:
2008-12-03 18:16:04 +01:00
property_ = properties.pop(0)
2006-03-21 22:33:56 +01:00
vcard_current_row += 1
vertical_fill = gtk.FILL
if not properties:
vertical_fill |= gtk.EXPAND
label = gtk.Label()
label.set_alignment(0, 0)
2008-12-03 18:16:04 +01:00
if property_[1]:
label.set_markup(property_[0])
vcard_table.attach(label, 1, 2, vcard_current_row,
vcard_current_row + 1, gtk.FILL, vertical_fill, 0, 0)
2006-03-21 22:33:56 +01:00
label = gtk.Label()
label.set_alignment(0, 0)
2008-12-03 18:16:04 +01:00
label.set_markup(property_[1])
2006-03-21 22:33:56 +01:00
label.set_line_wrap(True)
vcard_table.attach(label, 2, 3, vcard_current_row,
vcard_current_row + 1, gtk.EXPAND | gtk.FILL,
vertical_fill, 0, 0)
2006-03-21 22:33:56 +01:00
else:
2008-12-03 18:16:04 +01:00
label.set_markup(property_[0])
2007-08-09 17:39:18 +02:00
label.set_line_wrap(True)
vcard_table.attach(label, 1, 3, vcard_current_row,
vcard_current_row + 1, gtk.FILL, vertical_fill, 0)
2008-12-03 22:56:12 +01:00
2006-03-21 22:33:56 +01:00
self.avatar_image.set_alignment(0, 0)
2008-12-03 22:56:12 +01:00
vcard_table.attach(self.avatar_image, 3, 4, 2, vcard_current_row + 1,
gtk.FILL, gtk.FILL | gtk.EXPAND, 3, 3)
2006-03-21 22:33:56 +01:00
self.win.add(vcard_table)
2005-08-24 14:58:03 +02:00
2005-11-09 22:05:55 +01:00
class RosterTooltip(NotificationAreaTooltip):
2009-11-25 21:59:43 +01:00
"""
Tooltip that is shown in the roster treeview
"""
def __init__(self):
self.account = None
self.image = gtk.Image()
self.image.set_alignment(0, 0)
2005-08-26 18:31:22 +02:00
# padding is independent of the total length and better than alignment
2008-12-03 22:56:12 +01:00
self.image.set_padding(1, 2)
self.avatar_image = gtk.Image()
2005-11-09 22:05:55 +01:00
NotificationAreaTooltip.__init__(self)
def populate(self, contacts):
self.create_window()
2007-06-03 12:04:20 +02:00
self.create_table()
2005-11-09 22:05:55 +01:00
if not contacts or len(contacts) == 0:
# Tooltip for merged accounts row
2007-08-09 17:39:18 +02:00
accounts = helpers.get_notification_icon_tooltip_dict()
2005-11-09 22:05:55 +01:00
self.table.resize(2, 1)
self.spacer_label = ''
self.fill_table_with_accounts(accounts)
2006-03-21 22:33:56 +01:00
self.win.add(self.table)
2005-11-09 22:05:55 +01:00
return
2008-12-03 22:56:12 +01:00
# primary contact
prim_contact = gajim.contacts.get_highest_prio_contact_from_contacts(
contacts)
2008-12-03 22:56:12 +01:00
puny_jid = helpers.sanitize_filename(prim_contact.jid)
table_size = 3
2007-06-03 12:04:20 +02:00
2008-12-03 18:16:04 +01:00
file_ = helpers.get_avatar_path(os.path.join(gajim.AVATAR_PATH, puny_jid))
if file_:
self.avatar_image.set_from_file(file_)
2007-06-03 12:04:20 +02:00
pix = self.avatar_image.get_pixbuf()
pix = gtkgui_helpers.get_scaled_pixbuf(pix, 'tooltip')
self.avatar_image.set_from_pixbuf(pix)
table_size = 4
else:
self.avatar_image.set_from_pixbuf(None)
vcard_table = gtk.Table(table_size, 1)
vcard_table.set_property('column-spacing', 2)
vcard_table.set_homogeneous(False)
vcard_current_row = 1
properties = []
2006-10-11 04:12:56 +02:00
name_markup = u'<span weight="bold">' + \
gobject.markup_escape_text(prim_contact.get_shown_name())\
+ '</span>'
if self.account and helpers.jid_is_blocked(self.account,
prim_contact.jid):
2007-06-03 12:04:20 +02:00
name_markup += _(' [blocked]')
if self.account and \
self.account in gajim.interface.minimized_controls and \
2007-06-03 12:04:20 +02:00
prim_contact.jid in gajim.interface.minimized_controls[self.account]:
name_markup += _(' [minimized]')
properties.append((name_markup, None))
2007-06-03 12:04:20 +02:00
num_resources = 0
2006-05-26 22:41:03 +02:00
# put contacts in dict, where key is priority
contacts_dict = {}
for contact in contacts:
if contact.resource:
num_resources += 1
2006-05-26 22:41:03 +02:00
if contact.priority in contacts_dict:
contacts_dict[contact.priority].append(contact)
else:
contacts_dict[contact.priority] = [contact]
if num_resources > 1:
properties.append((_('Status: '), ' '))
2006-09-28 03:51:32 +02:00
transport = gajim.get_transport_name_from_jid(
prim_contact.jid)
2006-09-28 03:47:30 +02:00
if transport:
2007-12-12 09:44:46 +01:00
file_path = os.path.join(helpers.get_transport_path(transport),
'16x16')
2006-09-28 03:47:30 +02:00
else:
iconset = gajim.config.get('iconset')
if not iconset:
iconset = 'dcraven'
2007-08-09 17:39:18 +02:00
file_path = os.path.join(helpers.get_iconset_path(iconset), '16x16')
2006-09-28 03:47:30 +02:00
contact_keys = sorted(contacts_dict.keys())
2006-05-31 08:36:49 +02:00
contact_keys.reverse()
2006-05-26 22:41:03 +02:00
for priority in contact_keys:
for acontact in contacts_dict[priority]:
status_line = self.get_status_info(acontact.resource,
acontact.priority, acontact.show, acontact.status)
2008-12-03 22:56:12 +01:00
icon_name = self._get_icon_name_for_tooltip(acontact)
self.add_status_row(file_path, icon_name, status_line,
acontact.last_status_time)
properties.append((self.table, None))
2007-08-10 01:05:43 +02:00
else: # only one resource
if contact.show:
2008-12-03 22:56:12 +01:00
show = helpers.get_uf_show(contact.show)
if not self.check_last_time and self.account:
if contact.show == 'offline':
if not contact.last_status_time:
gajim.connections[self.account].request_last_status_time(
contact.jid, '')
else:
self.check_last_time = contact.last_status_time
elif contact.resource:
gajim.connections[self.account].request_last_status_time(
contact.jid, contact.resource)
if contact.last_activity_time:
self.check_last_time = contact.last_activity_time
else:
self.check_last_time = None
if contact.last_status_time:
vcard_current_row += 1
if contact.show == 'offline':
text = ' - ' + _('Last status: %s')
else:
text = _(' since %s')
2008-12-03 22:56:12 +01:00
if time.strftime('%j', time.localtime())== \
time.strftime('%j', contact.last_status_time):
# it's today, show only the locale hour representation
local_time = time.strftime('%X',
contact.last_status_time)
else:
# time.strftime returns locale encoded string
local_time = time.strftime('%c',
contact.last_status_time)
local_time = local_time.decode(
locale.getpreferredencoding())
2008-12-03 22:56:12 +01:00
text = text % local_time
show += text
2007-06-03 12:04:20 +02:00
if self.account and \
prim_contact.jid in gajim.gc_connected[self.account]:
2007-06-03 12:04:20 +02:00
if gajim.gc_connected[self.account][prim_contact.jid]:
show = _('Connected')
else:
show = _('Disconnected')
show = '<i>' + show + '</i>'
# we append show below
2008-12-03 22:56:12 +01:00
if contact.status:
status = contact.status.strip()
if status:
2005-09-04 03:40:17 +02:00
# reduce long status
2007-06-03 12:04:20 +02:00
# (no more than 300 chars on line and no more than 5 lines)
# status is wrapped
status = helpers.reduce_chars_newlines(status, 300, 5)
2008-12-03 22:56:12 +01:00
# escape markup entities.
status = gobject.markup_escape_text(status)
properties.append(('<i>%s</i>' % status, None))
properties.append((show, None))
2008-12-03 22:56:12 +01:00
self._append_pep_info(contact, properties)
2008-12-03 22:56:12 +01:00
properties.append((_('Jabber ID: '), prim_contact.jid ))
# contact has only one ressource
if num_resources == 1 and contact.resource:
properties.append((_('Resource: '),
gobject.markup_escape_text(contact.resource) +\
' (' + unicode(contact.priority) + ')'))
2008-12-03 22:56:12 +01:00
2007-06-03 12:04:20 +02:00
if self.account and prim_contact.sub and prim_contact.sub != 'both' and\
prim_contact.jid not in gajim.gc_connected[self.account]:
# ('both' is the normal sub so we don't show it)
2008-12-03 22:56:12 +01:00
properties.append(( _('Subscription: '),
gobject.markup_escape_text(helpers.get_uf_sub(prim_contact.sub))))
2008-12-03 22:56:12 +01:00
if prim_contact.keyID:
keyID = None
if len(prim_contact.keyID) == 8:
keyID = prim_contact.keyID
elif len(prim_contact.keyID) == 16:
keyID = prim_contact.keyID[8:]
if keyID:
properties.append((_('OpenPGP: '),
gobject.markup_escape_text(keyID)))
2008-12-03 22:56:12 +01:00
if contact.last_activity_time:
text = _(' since %s')
if time.strftime('%j', time.localtime())== \
time.strftime('%j', contact.last_activity_time):
# it's today, show only the locale hour representation
local_time = time.strftime('%I:%M %p',
contact.last_activity_time)
else:
# time.strftime returns locale encoded string
local_time = time.strftime('%c',
contact.last_activity_time)
local_time = local_time.decode(
locale.getpreferredencoding())
text = text % local_time
properties.append(('Idle' + text,None))
while properties:
2008-12-03 18:16:04 +01:00
property_ = properties.pop(0)
vcard_current_row += 1
vertical_fill = gtk.FILL
if not properties and table_size == 4:
vertical_fill |= gtk.EXPAND
label = gtk.Label()
label.set_alignment(0, 0)
2008-12-03 18:16:04 +01:00
if property_[1]:
label.set_markup(property_[0])
vcard_table.attach(label, 1, 2, vcard_current_row,
2007-12-12 09:44:46 +01:00
vcard_current_row + 1, gtk.FILL, vertical_fill, 0, 0)
2006-03-21 16:44:36 +01:00
label = gtk.Label()
label.set_alignment(0, 0)
2008-12-03 18:16:04 +01:00
label.set_markup(property_[1])
label.set_line_wrap(True)
vcard_table.attach(label, 2, 3, vcard_current_row,
vcard_current_row + 1, gtk.EXPAND | gtk.FILL,
vertical_fill, 0, 0)
else:
2009-11-25 21:59:43 +01:00
if isinstance(property_[0], (unicode, str)): # FIXME: rm unicode?
2008-12-03 22:56:12 +01:00
label.set_markup(property_[0])
2007-06-03 12:04:20 +02:00
label.set_line_wrap(True)
else:
2008-12-03 18:16:04 +01:00
label = property_[0]
vcard_table.attach(label, 1, 3, vcard_current_row,
vcard_current_row + 1, gtk.FILL, vertical_fill, 0)
self.avatar_image.set_alignment(0, 0)
if table_size == 4:
vcard_table.attach(self.avatar_image, 3, 4, 2,
vcard_current_row + 1, gtk.FILL, gtk.FILL | gtk.EXPAND, 3, 3)
2006-03-21 22:33:56 +01:00
self.win.add(vcard_table)
def update_last_time(self, last_time):
if not self.check_last_time or time.strftime('%x %I:%M %p', last_time) !=\
time.strftime('%x %I:%M %p', self.check_last_time):
self.win.destroy()
self.win = None
self.populate(self.cur_data)
self.win.ensure_style()
self.win.show_all()
def _append_pep_info(self, contact, properties):
2009-11-25 21:59:43 +01:00
"""
Append Tune, Mood, Activity, Location information of the specified contact
to the given property list.
2009-11-25 21:59:43 +01:00
"""
if 'mood' in contact.pep:
mood = contact.pep['mood'].asMarkupText()
mood_string = _('Mood:') + ' %s' % mood
properties.append((mood_string, None))
if 'activity' in contact.pep:
activity = contact.pep['activity'].asMarkupText()
2009-11-25 21:59:43 +01:00
activity_string = _('Activity:') + ' %s' % activity
properties.append((activity_string, None))
if 'tune' in contact.pep:
tune = contact.pep['tune'].asMarkupText()
tune_string = _('Tune:') + ' %s' % tune
properties.append((tune_string, None))
2008-12-03 22:56:12 +01:00
if 'location' in contact.pep:
location = contact.pep['location'].asMarkupText()
location_string = _('Location:') + ' %s' % location
properties.append((location_string, None))
class FileTransfersTooltip(BaseTooltip):
2009-11-25 21:59:43 +01:00
"""
Tooltip that is shown in the notification area
"""
def __init__(self):
BaseTooltip.__init__(self)
def populate(self, file_props):
2006-03-21 18:01:23 +01:00
ft_table = gtk.Table(2, 1)
ft_table.set_property('column-spacing', 2)
current_row = 1
self.create_window()
2006-03-21 18:01:23 +01:00
properties = []
name = file_props['name']
if file_props['type'] == 'r':
2008-12-02 16:53:23 +01:00
file_name = os.path.split(file_props['file-name'])[1]
else:
file_name = file_props['name']
2008-12-03 22:56:12 +01:00
properties.append((_('Name: '),
gobject.markup_escape_text(file_name)))
if file_props['type'] == 'r':
2008-12-03 18:16:04 +01:00
type_ = _('Download')
2008-12-03 22:56:12 +01:00
actor = _('Sender: ')
sender = unicode(file_props['sender']).split('/')[0]
2008-12-03 22:56:12 +01:00
name = gajim.contacts.get_first_contact_from_jid(
file_props['tt_account'], sender).get_shown_name()
else:
2008-12-03 18:16:04 +01:00
type_ = _('Upload')
2006-03-21 18:01:23 +01:00
actor = _('Recipient: ')
receiver = file_props['receiver']
if hasattr(receiver, 'name'):
name = receiver.get_shown_name()
else:
name = receiver.split('/')[0]
2008-12-03 18:16:04 +01:00
properties.append((_('Type: '), type_))
properties.append((actor, gobject.markup_escape_text(name)))
2008-12-03 22:56:12 +01:00
transfered_len = file_props.get('received-len', 0)
2006-03-21 18:01:23 +01:00
properties.append((_('Transferred: '), helpers.convert_bytes(transfered_len)))
2008-12-03 22:56:12 +01:00
status = ''
if 'started' not in file_props or not file_props['started']:
2007-12-12 09:44:46 +01:00
status = _('Not started')
elif 'connected' in file_props:
if 'stopped' in file_props and \
file_props['stopped'] == True:
status = _('Stopped')
elif file_props['completed']:
2007-12-12 09:44:46 +01:00
status = _('Completed')
elif file_props['connected'] == False:
if file_props['completed']:
status = _('Completed')
else:
if 'paused' in file_props and \
file_props['paused'] == True:
status = _('?transfer status:Paused')
elif 'stalled' in file_props and \
file_props['stalled'] == True:
#stalled is not paused. it is like 'frozen' it stopped alone
status = _('Stalled')
else:
status = _('Transferring')
else:
2007-12-12 09:44:46 +01:00
status = _('Not started')
2006-03-21 18:01:23 +01:00
properties.append((_('Status: '), status))
2007-12-12 09:44:46 +01:00
if 'desc' in file_props:
file_desc = file_props['desc']
properties.append((_('Description: '), gobject.markup_escape_text(
file_desc)))
2006-03-21 18:01:23 +01:00
while properties:
2008-12-03 18:16:04 +01:00
property_ = properties.pop(0)
2006-03-21 18:01:23 +01:00
current_row += 1
label = gtk.Label()
label.set_alignment(0, 0)
2008-12-03 18:16:04 +01:00
label.set_markup(property_[0])
2008-12-03 22:56:12 +01:00
ft_table.attach(label, 1, 2, current_row, current_row + 1,
2007-12-12 09:44:46 +01:00
gtk.FILL, gtk.FILL, 0, 0)
2006-03-21 18:01:23 +01:00
label = gtk.Label()
label.set_alignment(0, 0)
label.set_line_wrap(True)
2008-12-03 18:16:04 +01:00
label.set_markup(property_[1])
2008-12-03 22:56:12 +01:00
ft_table.attach(label, 2, 3, current_row, current_row + 1,
2007-01-31 12:31:07 +01:00
gtk.EXPAND | gtk.FILL, gtk.FILL, 0, 0)
2008-12-03 22:56:12 +01:00
2006-03-21 18:01:23 +01:00
self.win.add(ft_table)
class ServiceDiscoveryTooltip(BaseTooltip):
2009-11-25 21:59:43 +01:00
"""
Tooltip that is shown when hovering over a service discovery row
"""
def populate(self, status):
self.create_window()
label = gtk.Label()
label.set_line_wrap(True)
label.set_alignment(0, 0)
label.set_selectable(False)
if status == 1:
2006-02-05 17:22:25 +01:00
label.set_text(
_('This service has not yet responded with detailed information'))
elif status == 2:
2006-02-05 17:22:25 +01:00
label.set_text(
_('This service could not respond with detailed information.\n'
'It is most likely legacy or broken'))
self.win.add(label)
2008-07-29 22:57:13 +02:00
# vim: se ts=3: