Refactor Roster Tooltip
This commit is contained in:
parent
ec99e93a7c
commit
3296c23e32
5 changed files with 762 additions and 352 deletions
|
@ -366,12 +366,9 @@
|
||||||
<signal name="button-press-event" handler="on_roster_treeview_button_press_event" swapped="no"/>
|
<signal name="button-press-event" handler="on_roster_treeview_button_press_event" swapped="no"/>
|
||||||
<signal name="button-release-event" handler="on_roster_treeview_button_release_event" swapped="no"/>
|
<signal name="button-release-event" handler="on_roster_treeview_button_release_event" swapped="no"/>
|
||||||
<signal name="key-press-event" handler="on_roster_treeview_key_press_event" swapped="no"/>
|
<signal name="key-press-event" handler="on_roster_treeview_key_press_event" swapped="no"/>
|
||||||
<signal name="leave-notify-event" handler="on_roster_treeview_leave_notify_event" swapped="no"/>
|
|
||||||
<signal name="motion-notify-event" handler="on_roster_treeview_motion_notify_event" swapped="no"/>
|
|
||||||
<signal name="row-activated" handler="on_roster_treeview_row_activated" swapped="no"/>
|
<signal name="row-activated" handler="on_roster_treeview_row_activated" swapped="no"/>
|
||||||
<signal name="row-collapsed" handler="on_roster_treeview_row_collapsed" swapped="no"/>
|
<signal name="row-collapsed" handler="on_roster_treeview_row_collapsed" swapped="no"/>
|
||||||
<signal name="row-expanded" handler="on_roster_treeview_row_expanded" swapped="no"/>
|
<signal name="row-expanded" handler="on_roster_treeview_row_expanded" swapped="no"/>
|
||||||
<signal name="scroll-event" handler="on_roster_treeview_scroll_event" swapped="no"/>
|
|
||||||
<child internal-child="selection">
|
<child internal-child="selection">
|
||||||
<object class="GtkTreeSelection" id="treeview-selection1"/>
|
<object class="GtkTreeSelection" id="treeview-selection1"/>
|
||||||
</child>
|
</child>
|
||||||
|
|
370
data/gui/tooltip_roster_contact.ui
Normal file
370
data/gui/tooltip_roster_contact.ui
Normal file
|
@ -0,0 +1,370 @@
|
||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<!-- Generated with glade 3.18.3 -->
|
||||||
|
<interface>
|
||||||
|
<requires lib="gtk+" version="3.12"/>
|
||||||
|
<object class="GtkGrid" id="tooltip_grid">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="column_spacing">5</property>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="name">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="valign">start</property>
|
||||||
|
<property name="use_markup">True</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">0</property>
|
||||||
|
<property name="width">2</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="jid_label">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="valign">start</property>
|
||||||
|
<property name="label" translatable="yes">Jabber ID:</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">4</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkImage" id="avatar">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="stock">gtk-missing-image</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">2</property>
|
||||||
|
<property name="top_attach">0</property>
|
||||||
|
<property name="height">14</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="resource_label">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="label" translatable="yes">Resource:</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">5</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="user_show">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="valign">start</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">1</property>
|
||||||
|
<property name="width">2</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="resource">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="hexpand">True</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">1</property>
|
||||||
|
<property name="top_attach">5</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="jid">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="hexpand">True</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
<attributes>
|
||||||
|
<attribute name="weight" value="bold"/>
|
||||||
|
</attributes>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">1</property>
|
||||||
|
<property name="top_attach">4</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="status_label">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="valign">start</property>
|
||||||
|
<property name="label" translatable="yes">Status:</property>
|
||||||
|
<property name="wrap">True</property>
|
||||||
|
<property name="max_width_chars">40</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">2</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="idle_since_label">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="label" translatable="yes">Idle since:</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">12</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="idle_for_label">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="label" translatable="yes">Idle for:</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">13</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="idle_since">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="hexpand">True</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">1</property>
|
||||||
|
<property name="top_attach">12</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="idle_for">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="hexpand">True</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">1</property>
|
||||||
|
<property name="top_attach">13</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="mood_label">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="label" translatable="yes">Mood:</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">6</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="activity_label">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="label" translatable="yes">Activity:</property>
|
||||||
|
<property name="lines">2</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">7</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="tune_label">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="label" translatable="yes">Tune:</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">8</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="location_label">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="label" translatable="yes">Location:</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">9</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="mood">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="hexpand">True</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">1</property>
|
||||||
|
<property name="top_attach">6</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="activity">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="hexpand">True</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">1</property>
|
||||||
|
<property name="top_attach">7</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="tune">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="hexpand">True</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">1</property>
|
||||||
|
<property name="top_attach">8</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="location">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="hexpand">True</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">1</property>
|
||||||
|
<property name="top_attach">9</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="pgp_label">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="label" translatable="yes">OpenPGP:</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">10</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="pgp">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="hexpand">True</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">1</property>
|
||||||
|
<property name="top_attach">10</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="sub_label">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="halign">start</property>
|
||||||
|
<property name="label" translatable="yes">Subscription:</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">0</property>
|
||||||
|
<property name="top_attach">11</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="sub">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="hexpand">True</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">1</property>
|
||||||
|
<property name="top_attach">11</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<object class="GtkLabel" id="status">
|
||||||
|
<property name="visible">True</property>
|
||||||
|
<property name="can_focus">False</property>
|
||||||
|
<property name="hexpand">True</property>
|
||||||
|
<property name="wrap">True</property>
|
||||||
|
<property name="max_width_chars">30</property>
|
||||||
|
<property name="xalign">0</property>
|
||||||
|
<property name="yalign">0</property>
|
||||||
|
<attributes>
|
||||||
|
<attribute name="style" value="italic"/>
|
||||||
|
</attributes>
|
||||||
|
</object>
|
||||||
|
<packing>
|
||||||
|
<property name="left_attach">1</property>
|
||||||
|
<property name="top_attach">2</property>
|
||||||
|
</packing>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<placeholder/>
|
||||||
|
</child>
|
||||||
|
<child>
|
||||||
|
<placeholder/>
|
||||||
|
</child>
|
||||||
|
</object>
|
||||||
|
</interface>
|
|
@ -562,11 +562,14 @@ class Interface:
|
||||||
|
|
||||||
def handle_event_last_status_time(self, obj):
|
def handle_event_last_status_time(self, obj):
|
||||||
# ('LAST_STATUS_TIME', account, (jid, resource, seconds, status))
|
# ('LAST_STATUS_TIME', account, (jid, resource, seconds, status))
|
||||||
if obj.seconds < 0:
|
|
||||||
# Ann error occured
|
|
||||||
return
|
|
||||||
account = obj.conn.name
|
account = obj.conn.name
|
||||||
c = gajim.contacts.get_contact(account, obj.jid, obj.resource)
|
c = gajim.contacts.get_contact(account, obj.jid, obj.resource)
|
||||||
|
tooltip_window = self.roster.tree.get_tooltip_window()
|
||||||
|
if obj.seconds < 0:
|
||||||
|
if tooltip_window:
|
||||||
|
tooltip_window.update_last_time(c, True)
|
||||||
|
return
|
||||||
|
|
||||||
if c: # c can be none if it's a gc contact
|
if c: # c can be none if it's a gc contact
|
||||||
if obj.status:
|
if obj.status:
|
||||||
c.status = obj.status
|
c.status = obj.status
|
||||||
|
@ -576,8 +579,10 @@ class Interface:
|
||||||
c.last_status_time = last_time
|
c.last_status_time = last_time
|
||||||
else:
|
else:
|
||||||
c.last_activity_time = last_time
|
c.last_activity_time = last_time
|
||||||
if self.roster.tooltip.id and self.roster.tooltip.win:
|
|
||||||
self.roster.tooltip.update_last_time(last_time)
|
# Set last time on roster tooltip
|
||||||
|
if tooltip_window:
|
||||||
|
tooltip_window.update_last_time(c)
|
||||||
|
|
||||||
def handle_event_gc_config(self, obj):
|
def handle_event_gc_config(self, obj):
|
||||||
#('GC_CONFIG', account, (jid, form_node)) config is a dict
|
#('GC_CONFIG', account, (jid, form_node)) config is a dict
|
||||||
|
|
|
@ -2063,25 +2063,6 @@ class RosterWindow:
|
||||||
vb.hide()
|
vb.hide()
|
||||||
vb.set_no_show_all(True)
|
vb.set_no_show_all(True)
|
||||||
|
|
||||||
def show_tooltip(self, contact):
|
|
||||||
self.tooltip.timeout = 0
|
|
||||||
device = self.tree.get_window().get_display().get_device_manager().\
|
|
||||||
get_client_pointer()
|
|
||||||
pointer = self.tree.get_window().get_device_position(device)
|
|
||||||
props = self.tree.get_path_at_pos(pointer[1], pointer[2])
|
|
||||||
# check if the current pointer is at the same path
|
|
||||||
# as it was before setting the timeout
|
|
||||||
if props and self.tooltip.id == props[0]:
|
|
||||||
# bounding rectangle of coordinates for the cell within the treeview
|
|
||||||
rect = self.tree.get_cell_area(props[0], props[1])
|
|
||||||
|
|
||||||
# position of the treeview on the screen
|
|
||||||
position = self.tree.get_window().get_origin()
|
|
||||||
self.tooltip.show_tooltip(contact, rect.height, position[2] + \
|
|
||||||
rect.y)
|
|
||||||
else:
|
|
||||||
self.tooltip.hide_tooltip()
|
|
||||||
|
|
||||||
def authorize(self, widget, jid, account):
|
def authorize(self, widget, jid, account):
|
||||||
"""
|
"""
|
||||||
Authorize a contact (by re-sending auth menuitem)
|
Authorize a contact (by re-sending auth menuitem)
|
||||||
|
@ -2443,7 +2424,6 @@ class RosterWindow:
|
||||||
if not gajim.config.get('quit_on_roster_x_button') and (
|
if not gajim.config.get('quit_on_roster_x_button') and (
|
||||||
(gajim.interface.systray_enabled and gajim.config.get('trayicon') != \
|
(gajim.interface.systray_enabled and gajim.config.get('trayicon') != \
|
||||||
'on_event') or gajim.config.get('allow_hide_roster')):
|
'on_event') or gajim.config.get('allow_hide_roster')):
|
||||||
self.tooltip.hide_tooltip()
|
|
||||||
if gajim.config.get('save-roster-position'):
|
if gajim.config.get('save-roster-position'):
|
||||||
x, y = self.window.get_position()
|
x, y = self.window.get_position()
|
||||||
gajim.config.set('roster_x-position', x)
|
gajim.config.set('roster_x-position', x)
|
||||||
|
@ -2894,119 +2874,6 @@ class RosterWindow:
|
||||||
return
|
return
|
||||||
info[contact.jid] = vcard.ZeroconfVcardWindow(contact, account)
|
info[contact.jid] = vcard.ZeroconfVcardWindow(contact, account)
|
||||||
|
|
||||||
def on_roster_treeview_leave_notify_event(self, widget, event):
|
|
||||||
props = widget.get_path_at_pos(int(event.x), int(event.y))
|
|
||||||
if self.tooltip.timeout > 0 or self.tooltip.shown:
|
|
||||||
if not props or self.tooltip.id == props[0]:
|
|
||||||
self.tooltip.hide_tooltip()
|
|
||||||
|
|
||||||
def on_roster_treeview_motion_notify_event(self, widget, event):
|
|
||||||
model = widget.get_model()
|
|
||||||
props = widget.get_path_at_pos(int(event.x), int(event.y))
|
|
||||||
if self.tooltip.timeout > 0 or self.tooltip.shown:
|
|
||||||
if not props or self.tooltip.id != props[0]:
|
|
||||||
self.tooltip.hide_tooltip()
|
|
||||||
if props:
|
|
||||||
row = props[0]
|
|
||||||
titer = None
|
|
||||||
try:
|
|
||||||
titer = model.get_iter(row)
|
|
||||||
except Exception:
|
|
||||||
self.tooltip.hide_tooltip()
|
|
||||||
return
|
|
||||||
if model[titer][C_TYPE] in ('contact', 'self_contact'):
|
|
||||||
# we're on a contact entry in the roster
|
|
||||||
if (not self.tooltip.shown and self.tooltip.timeout == 0) or \
|
|
||||||
self.tooltip.id != props[0]:
|
|
||||||
account = model[titer][C_ACCOUNT]
|
|
||||||
jid = model[titer][C_JID]
|
|
||||||
self.tooltip.id = row
|
|
||||||
contacts = gajim.contacts.get_contacts(account, jid)
|
|
||||||
connected_contacts = []
|
|
||||||
for c in contacts:
|
|
||||||
if c.show not in ('offline', 'error'):
|
|
||||||
connected_contacts.append(c)
|
|
||||||
if not connected_contacts:
|
|
||||||
# no connected contacts, show the ofline one
|
|
||||||
connected_contacts = contacts
|
|
||||||
self.tooltip.account = account
|
|
||||||
self.tooltip.timeout = GLib.timeout_add(500,
|
|
||||||
self.show_tooltip, connected_contacts)
|
|
||||||
elif model[titer][C_TYPE] == 'groupchat':
|
|
||||||
if (not self.tooltip.shown and self.tooltip.timeout == 0) or \
|
|
||||||
self.tooltip.id != props[0]:
|
|
||||||
account = model[titer][C_ACCOUNT]
|
|
||||||
jid = model[titer][C_JID]
|
|
||||||
self.tooltip.id = row
|
|
||||||
contact = gajim.contacts.get_contacts(account, jid)
|
|
||||||
self.tooltip.account = account
|
|
||||||
self.tooltip.timeout = GLib.timeout_add(500,
|
|
||||||
self.show_tooltip, contact)
|
|
||||||
elif model[titer][C_TYPE] == 'account':
|
|
||||||
# we're on an account entry in the roster
|
|
||||||
if (not self.tooltip.shown and self.tooltip.timeout == 0) or \
|
|
||||||
self.tooltip.id != props[0]:
|
|
||||||
account = model[titer][C_ACCOUNT]
|
|
||||||
if account == 'all':
|
|
||||||
self.tooltip.id = row
|
|
||||||
self.tooltip.account = None
|
|
||||||
self.tooltip.timeout = GLib.timeout_add(500,
|
|
||||||
self.show_tooltip, [])
|
|
||||||
return
|
|
||||||
jid = gajim.get_jid_from_account(account)
|
|
||||||
contacts = []
|
|
||||||
connection = gajim.connections[account]
|
|
||||||
# get our current contact info
|
|
||||||
|
|
||||||
nbr_on, nbr_total = gajim.\
|
|
||||||
contacts.get_nb_online_total_contacts(
|
|
||||||
accounts=[account])
|
|
||||||
account_name = account
|
|
||||||
if gajim.account_is_connected(account):
|
|
||||||
account_name += ' (%s/%s)' % (repr(nbr_on),
|
|
||||||
repr(nbr_total))
|
|
||||||
contact = gajim.contacts.create_self_contact(jid=jid,
|
|
||||||
account=account, name=account_name,
|
|
||||||
show=connection.get_status(), status=connection.status,
|
|
||||||
resource=connection.server_resource,
|
|
||||||
priority=connection.priority)
|
|
||||||
if gajim.connections[account].gpg:
|
|
||||||
contact.keyID = gajim.config.get_per('accounts',
|
|
||||||
connection.name, 'keyid')
|
|
||||||
contacts.append(contact)
|
|
||||||
# if we're online ...
|
|
||||||
if connection.connection:
|
|
||||||
roster = connection.connection.getRoster()
|
|
||||||
# in threadless connection when no roster stanza is sent
|
|
||||||
# 'roster' is None
|
|
||||||
if roster and roster.getItem(jid):
|
|
||||||
resources = roster.getResources(jid)
|
|
||||||
# ...get the contact info for our other online
|
|
||||||
# resources
|
|
||||||
for resource in resources:
|
|
||||||
# Check if we already have this resource
|
|
||||||
found = False
|
|
||||||
for contact_ in contacts:
|
|
||||||
if contact_.resource == resource:
|
|
||||||
found = True
|
|
||||||
break
|
|
||||||
if found:
|
|
||||||
continue
|
|
||||||
show = roster.getShow(jid+'/'+resource)
|
|
||||||
if not show:
|
|
||||||
show = 'online'
|
|
||||||
contact = gajim.contacts.create_self_contact(
|
|
||||||
jid=jid, account=account, show=show,
|
|
||||||
status=roster.getStatus(
|
|
||||||
jid + '/' + resource),
|
|
||||||
priority=roster.getPriority(
|
|
||||||
jid + '/' + resource), resource=resource)
|
|
||||||
contacts.append(contact)
|
|
||||||
self.tooltip.id = row
|
|
||||||
self.tooltip.account = None
|
|
||||||
self.tooltip.timeout = GLib.timeout_add(500,
|
|
||||||
self.show_tooltip, contacts)
|
|
||||||
|
|
||||||
def on_agent_logging(self, widget, jid, state, account):
|
def on_agent_logging(self, widget, jid, state, account):
|
||||||
"""
|
"""
|
||||||
When an agent is requested to log in or off
|
When an agent is requested to log in or off
|
||||||
|
@ -3449,15 +3316,10 @@ class RosterWindow:
|
||||||
def on_add_to_roster(self, widget, contact, account):
|
def on_add_to_roster(self, widget, contact, account):
|
||||||
dialogs.AddNewContactWindow(account, contact.jid, contact.name)
|
dialogs.AddNewContactWindow(account, contact.jid, contact.name)
|
||||||
|
|
||||||
|
|
||||||
def on_roster_treeview_scroll_event(self, widget, event):
|
|
||||||
self.tooltip.hide_tooltip()
|
|
||||||
|
|
||||||
def on_roster_treeview_key_press_event(self, widget, event):
|
def on_roster_treeview_key_press_event(self, widget, event):
|
||||||
"""
|
"""
|
||||||
When a key is pressed in the treeviews
|
When a key is pressed in the treeviews
|
||||||
"""
|
"""
|
||||||
self.tooltip.hide_tooltip()
|
|
||||||
if event.keyval == Gdk.KEY_Escape:
|
if event.keyval == Gdk.KEY_Escape:
|
||||||
if self.rfilter_enabled:
|
if self.rfilter_enabled:
|
||||||
self.disable_rfilter()
|
self.disable_rfilter()
|
||||||
|
@ -3577,8 +3439,6 @@ class RosterWindow:
|
||||||
self.enable_rfilter('')
|
self.enable_rfilter('')
|
||||||
|
|
||||||
def on_roster_treeview_button_press_event(self, widget, event):
|
def on_roster_treeview_button_press_event(self, widget, event):
|
||||||
# hide tooltip, no matter the button is pressed
|
|
||||||
self.tooltip.hide_tooltip()
|
|
||||||
try:
|
try:
|
||||||
pos = self.tree.get_path_at_pos(int(event.x), int(event.y))
|
pos = self.tree.get_path_at_pos(int(event.x), int(event.y))
|
||||||
path, x = pos[0], pos[2]
|
path, x = pos[0], pos[2]
|
||||||
|
@ -4079,7 +3939,6 @@ class RosterWindow:
|
||||||
'quit_on_roster_x_button') and ((gajim.interface.systray_enabled and\
|
'quit_on_roster_x_button') and ((gajim.interface.systray_enabled and\
|
||||||
gajim.config.get('trayicon') == 'always') or gajim.config.get(
|
gajim.config.get('trayicon') == 'always') or gajim.config.get(
|
||||||
'allow_hide_roster')):
|
'allow_hide_roster')):
|
||||||
self.tooltip.hide_tooltip()
|
|
||||||
self.window.hide()
|
self.window.hide()
|
||||||
elif event.get_state() & Gdk.ModifierType.CONTROL_MASK and event.keyval == \
|
elif event.get_state() & Gdk.ModifierType.CONTROL_MASK and event.keyval == \
|
||||||
Gdk.KEY_i:
|
Gdk.KEY_i:
|
||||||
|
@ -6268,6 +6127,54 @@ class RosterWindow:
|
||||||
renderer.set_property(self.renderers_propertys[renderer][0],
|
renderer.set_property(self.renderers_propertys[renderer][0],
|
||||||
self.renderers_propertys[renderer][1])
|
self.renderers_propertys[renderer][1])
|
||||||
|
|
||||||
|
def query_tooltip(self, widget, x_pos, y_pos, keyboard_mode, tooltip):
|
||||||
|
try:
|
||||||
|
row = widget.get_path_at_pos(x_pos, y_pos)[0]
|
||||||
|
except TypeError:
|
||||||
|
return False
|
||||||
|
if not row:
|
||||||
|
return False
|
||||||
|
|
||||||
|
iter_ = None
|
||||||
|
try:
|
||||||
|
model = widget.get_model()
|
||||||
|
iter_ = model.get_iter(row)
|
||||||
|
except Exception:
|
||||||
|
return False
|
||||||
|
|
||||||
|
typ = model[iter_][C_TYPE]
|
||||||
|
account = model[iter_][C_ACCOUNT]
|
||||||
|
jid = model[iter_][C_JID]
|
||||||
|
connected_contacts = []
|
||||||
|
|
||||||
|
if typ in ('contact', 'self_contact'):
|
||||||
|
contacts = gajim.contacts.get_contacts(account, jid)
|
||||||
|
|
||||||
|
for c in contacts:
|
||||||
|
if c.show not in ('offline', 'error'):
|
||||||
|
connected_contacts.append(c)
|
||||||
|
if not connected_contacts:
|
||||||
|
# no connected contacts, show the offline one
|
||||||
|
connected_contacts = contacts
|
||||||
|
elif typ == 'groupchat':
|
||||||
|
connected_contacts = gajim.contacts.get_contacts(account, jid)
|
||||||
|
elif typ != 'account':
|
||||||
|
return False
|
||||||
|
|
||||||
|
if self.current_tooltip != row:
|
||||||
|
# If the row changes we hide the current tooltip
|
||||||
|
self.current_tooltip = row
|
||||||
|
return False
|
||||||
|
|
||||||
|
tooltip = widget.get_tooltip_window()
|
||||||
|
|
||||||
|
if tooltip.row == row:
|
||||||
|
# We already populated the window with the row data
|
||||||
|
return True
|
||||||
|
tooltip.row = row
|
||||||
|
tooltip.populate(connected_contacts, account, typ)
|
||||||
|
return True
|
||||||
|
|
||||||
################################################################################
|
################################################################################
|
||||||
###
|
###
|
||||||
################################################################################
|
################################################################################
|
||||||
|
@ -6515,7 +6422,10 @@ class RosterWindow:
|
||||||
self.combobox_callback_active = True
|
self.combobox_callback_active = True
|
||||||
|
|
||||||
self.collapsed_rows = gajim.config.get('collapsed_rows').split('\t')
|
self.collapsed_rows = gajim.config.get('collapsed_rows').split('\t')
|
||||||
self.tooltip = tooltips.RosterTooltip()
|
self.tree.set_has_tooltip(True)
|
||||||
|
self.tree.set_tooltip_window(tooltips.RosterTooltip(self.window))
|
||||||
|
self.current_tooltip = None
|
||||||
|
self.tree.connect('query-tooltip', self.query_tooltip)
|
||||||
# Workaroung: For strange reasons signal is behaving like row-changed
|
# Workaroung: For strange reasons signal is behaving like row-changed
|
||||||
self._toggeling_row = False
|
self._toggeling_row = False
|
||||||
self.setup_and_draw_roster()
|
self.setup_and_draw_roster()
|
||||||
|
|
532
src/tooltips.py
532
src/tooltips.py
|
@ -195,28 +195,28 @@ class StatusTable:
|
||||||
"""
|
"""
|
||||||
|
|
||||||
def __init__(self):
|
def __init__(self):
|
||||||
self.current_row = 1
|
self.current_row = 0
|
||||||
self.table = None
|
self.table = None
|
||||||
self.text_label = None
|
self.text_label = None
|
||||||
self.spacer_label = ' '
|
self.spacer_label = ' '
|
||||||
|
|
||||||
def create_table(self):
|
def create_table(self):
|
||||||
self.table = Gtk.Grid()
|
self.table = Gtk.Grid()
|
||||||
self.table.insert_row(0)
|
|
||||||
self.table.insert_row(0)
|
|
||||||
self.table.insert_column(0)
|
self.table.insert_column(0)
|
||||||
self.table.set_property('column-spacing', 2)
|
self.table.set_property('column-spacing', 2)
|
||||||
|
|
||||||
def add_text_row(self, text, col_inc = 0):
|
def add_text_row(self, text, col_inc=0):
|
||||||
self.current_row += 1
|
self.table.insert_row(self.current_row)
|
||||||
self.text_label = Gtk.Label()
|
self.text_label = Gtk.Label()
|
||||||
self.text_label.set_line_wrap(True)
|
self.text_label.set_line_wrap(True)
|
||||||
|
self.text_label.set_max_width_chars(35)
|
||||||
self.text_label.set_halign(Gtk.Align.START)
|
self.text_label.set_halign(Gtk.Align.START)
|
||||||
self.text_label.set_valign(Gtk.Align.START)
|
self.text_label.set_valign(Gtk.Align.START)
|
||||||
self.text_label.set_selectable(False)
|
self.text_label.set_selectable(False)
|
||||||
self.text_label.set_markup(text)
|
self.text_label.set_markup(text)
|
||||||
self.table.attach(self.text_label, 1 + col_inc, self.current_row,
|
self.table.attach(self.text_label, 1 + col_inc, self.current_row,
|
||||||
3 - col_inc, 1)
|
3 - col_inc, 1)
|
||||||
|
self.current_row += 1
|
||||||
|
|
||||||
def get_status_info(self, resource, priority, show, status):
|
def get_status_info(self, resource, priority, show, status):
|
||||||
str_status = resource + ' (' + str(priority) + ')'
|
str_status = resource + ' (' + str(priority) + ')'
|
||||||
|
@ -235,9 +235,7 @@ class StatusTable:
|
||||||
"""
|
"""
|
||||||
Append a new row with status icon to the table
|
Append a new row with status icon to the table
|
||||||
"""
|
"""
|
||||||
self.table.insert_row(0)
|
self.table.insert_row(self.current_row)
|
||||||
self.table.insert_row(0)
|
|
||||||
self.current_row += 1
|
|
||||||
state_file = show.replace(' ', '_')
|
state_file = show.replace(' ', '_')
|
||||||
files = []
|
files = []
|
||||||
files.append(os.path.join(file_path, state_file + '.png'))
|
files.append(os.path.join(file_path, state_file + '.png'))
|
||||||
|
@ -265,6 +263,7 @@ class StatusTable:
|
||||||
lock_image.set_from_stock(Gtk.STOCK_DIALOG_AUTHENTICATION,
|
lock_image.set_from_stock(Gtk.STOCK_DIALOG_AUTHENTICATION,
|
||||||
Gtk.IconSize.MENU)
|
Gtk.IconSize.MENU)
|
||||||
self.table.attach(lock_image, 4, self.current_row, 1, 1)
|
self.table.attach(lock_image, 4, self.current_row, 1, 1)
|
||||||
|
self.current_row += 1
|
||||||
|
|
||||||
class NotificationAreaTooltip(BaseTooltip, StatusTable):
|
class NotificationAreaTooltip(BaseTooltip, StatusTable):
|
||||||
"""
|
"""
|
||||||
|
@ -414,83 +413,181 @@ class GCTooltip(Gtk.Window):
|
||||||
affiliation = formatted % (color, affiliation)
|
affiliation = formatted % (color, affiliation)
|
||||||
return affiliation
|
return affiliation
|
||||||
|
|
||||||
class RosterTooltip(NotificationAreaTooltip):
|
class RosterTooltip(Gtk.Window, StatusTable):
|
||||||
"""
|
# pylint: disable=E1101
|
||||||
Tooltip that is shown in the roster treeview
|
def __init__(self, parent):
|
||||||
"""
|
Gtk.Window.__init__(self, type=Gtk.WindowType.POPUP, transient_for=parent)
|
||||||
|
StatusTable.__init__(self)
|
||||||
def __init__(self):
|
|
||||||
self.account = None
|
|
||||||
self.avatar_image = Gtk.Image()
|
|
||||||
NotificationAreaTooltip.__init__(self)
|
|
||||||
|
|
||||||
def populate(self, contacts):
|
|
||||||
self.create_window()
|
|
||||||
|
|
||||||
self.create_table()
|
self.create_table()
|
||||||
if not contacts or len(contacts) == 0:
|
self.row = None
|
||||||
|
self.check_last_time = {}
|
||||||
|
self.contact_jid = None
|
||||||
|
self.last_widget = None
|
||||||
|
self.num_resources = 0
|
||||||
|
self.set_title('tooltip')
|
||||||
|
self.set_border_width(3)
|
||||||
|
self.set_resizable(False)
|
||||||
|
self.set_name('gtk-tooltips')
|
||||||
|
self.set_type_hint(Gdk.WindowTypeHint.TOOLTIP)
|
||||||
|
|
||||||
|
self.xml = gtkgui_helpers.get_gtk_builder('tooltip_roster_contact.ui')
|
||||||
|
for name in ('name', 'status', 'jid', 'user_show', 'fillelement',
|
||||||
|
'resource', 'avatar', 'resource_label', 'pgp', 'pgp_label',
|
||||||
|
'jid_label', 'tooltip_grid', 'idle_since', 'idle_for',
|
||||||
|
'idle_since_label', 'idle_for_label', 'mood', 'tune',
|
||||||
|
'activity', 'location', 'tune_label', 'location_label',
|
||||||
|
'activity_label', 'mood_label', 'sub_label', 'sub',
|
||||||
|
'status_label'):
|
||||||
|
setattr(self, name, self.xml.get_object(name))
|
||||||
|
|
||||||
|
self.add(self.tooltip_grid)
|
||||||
|
self.tooltip_grid.show()
|
||||||
|
|
||||||
|
def clear_tooltip(self):
|
||||||
|
"""
|
||||||
|
Hide all Elements of the Tooltip Grid
|
||||||
|
"""
|
||||||
|
for child in self.tooltip_grid.get_children():
|
||||||
|
child.hide()
|
||||||
|
status_table = self.tooltip_grid.get_child_at(0, 3)
|
||||||
|
if status_table:
|
||||||
|
status_table.destroy()
|
||||||
|
self.create_table()
|
||||||
|
|
||||||
|
def fill_table_with_accounts(self, accounts):
|
||||||
|
iconset = gajim.config.get('iconset')
|
||||||
|
if not iconset:
|
||||||
|
iconset = 'dcraven'
|
||||||
|
file_path = os.path.join(helpers.get_iconset_path(iconset), '16x16')
|
||||||
|
for acct in accounts:
|
||||||
|
message = acct['message']
|
||||||
|
message = helpers.reduce_chars_newlines(message, 100, 1)
|
||||||
|
message = GLib.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
|
||||||
|
if message:
|
||||||
|
self.add_status_row(file_path, acct['show'],
|
||||||
|
GLib.markup_escape_text(acct['name']) + ' - ' + message,
|
||||||
|
show_lock=show_lock, indent=False)
|
||||||
|
else:
|
||||||
|
self.add_status_row(file_path, acct['show'],
|
||||||
|
GLib.markup_escape_text(acct['name']), show_lock=show_lock,
|
||||||
|
indent=False)
|
||||||
|
for line in acct['event_lines']:
|
||||||
|
self.add_text_row(' ' + line, 1)
|
||||||
|
|
||||||
|
def populate(self, contacts, account, typ):
|
||||||
|
"""
|
||||||
|
Populate the Tooltip Grid with data of from the contact
|
||||||
|
"""
|
||||||
|
self.current_row = 0
|
||||||
|
self.account = account
|
||||||
|
if self.last_widget:
|
||||||
|
self.last_widget.set_vexpand(False)
|
||||||
|
|
||||||
|
self.clear_tooltip()
|
||||||
|
|
||||||
|
if account == 'all':
|
||||||
# Tooltip for merged accounts row
|
# Tooltip for merged accounts row
|
||||||
accounts = helpers.get_notification_icon_tooltip_dict()
|
accounts = helpers.get_notification_icon_tooltip_dict()
|
||||||
self.spacer_label = ''
|
self.spacer_label = ''
|
||||||
self.fill_table_with_accounts(accounts)
|
self.fill_table_with_accounts(accounts)
|
||||||
self.win.add(self.table)
|
self.tooltip_grid.attach(self.table, 0, 3, 2, 1)
|
||||||
|
self.table.show_all()
|
||||||
return
|
return
|
||||||
|
|
||||||
# primary contact
|
if typ == 'account':
|
||||||
prim_contact = gajim.contacts.get_highest_prio_contact_from_contacts(
|
jid = gajim.get_jid_from_account(account)
|
||||||
|
contacts = []
|
||||||
|
connection = gajim.connections[account]
|
||||||
|
# get our current contact info
|
||||||
|
|
||||||
|
nbr_on, nbr_total = gajim.\
|
||||||
|
contacts.get_nb_online_total_contacts(
|
||||||
|
accounts=[account])
|
||||||
|
account_name = account
|
||||||
|
if gajim.account_is_connected(account):
|
||||||
|
account_name += ' (%s/%s)' % (repr(nbr_on),
|
||||||
|
repr(nbr_total))
|
||||||
|
contact = gajim.contacts.create_self_contact(jid=jid,
|
||||||
|
account=account, name=account_name,
|
||||||
|
show=connection.get_status(), status=connection.status,
|
||||||
|
resource=connection.server_resource,
|
||||||
|
priority=connection.priority)
|
||||||
|
if gajim.connections[account].gpg:
|
||||||
|
contact.keyID = gajim.config.get_per('accounts',
|
||||||
|
connection.name, 'keyid')
|
||||||
|
contacts.append(contact)
|
||||||
|
# if we're online ...
|
||||||
|
if connection.connection:
|
||||||
|
roster = connection.connection.getRoster()
|
||||||
|
# in threadless connection when no roster stanza is sent
|
||||||
|
# 'roster' is None
|
||||||
|
if roster and roster.getItem(jid):
|
||||||
|
resources = roster.getResources(jid)
|
||||||
|
# ...get the contact info for our other online
|
||||||
|
# resources
|
||||||
|
for resource in resources:
|
||||||
|
# Check if we already have this resource
|
||||||
|
found = False
|
||||||
|
for contact_ in contacts:
|
||||||
|
if contact_.resource == resource:
|
||||||
|
found = True
|
||||||
|
break
|
||||||
|
if found:
|
||||||
|
continue
|
||||||
|
show = roster.getShow(jid + '/' + resource)
|
||||||
|
if not show:
|
||||||
|
show = 'online'
|
||||||
|
contact = gajim.contacts.create_self_contact(
|
||||||
|
jid=jid, account=account, show=show,
|
||||||
|
status=roster.getStatus(
|
||||||
|
jid + '/' + resource),
|
||||||
|
priority=roster.getPriority(
|
||||||
|
jid + '/' + resource), resource=resource)
|
||||||
|
contacts.append(contact)
|
||||||
|
|
||||||
|
# Username/Account/Groupchat
|
||||||
|
self.prim_contact = gajim.contacts.get_highest_prio_contact_from_contacts(
|
||||||
contacts)
|
contacts)
|
||||||
|
self.contact_jid = self.prim_contact.jid
|
||||||
puny_jid = helpers.sanitize_filename(prim_contact.jid)
|
name = GLib.markup_escape_text(self.prim_contact.get_shown_name())
|
||||||
table_size = 3
|
name_markup = '<b>{}</b>'.format(name)
|
||||||
|
|
||||||
file_ = helpers.get_avatar_path(os.path.join(gajim.AVATAR_PATH,
|
|
||||||
puny_jid))
|
|
||||||
if file_:
|
|
||||||
with open(file_, 'rb') as file_data:
|
|
||||||
pix = gtkgui_helpers.get_pixbuf_from_data(file_data.read())
|
|
||||||
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.Grid()
|
|
||||||
vcard_table.insert_row(0)
|
|
||||||
for i in range(0, table_size):
|
|
||||||
vcard_table.insert_column(0)
|
|
||||||
vcard_table.set_property('column-spacing', 2)
|
|
||||||
vcard_current_row = 1
|
|
||||||
properties = []
|
|
||||||
|
|
||||||
name_markup = '<span weight="bold">' + GLib.markup_escape_text(
|
|
||||||
prim_contact.get_shown_name()) + '</span>'
|
|
||||||
if gajim.config.get('mergeaccounts'):
|
if gajim.config.get('mergeaccounts'):
|
||||||
name_markup += " <span foreground='%s'>(%s)</span>" % (
|
color = gajim.config.get('tooltip_account_name_color')
|
||||||
gajim.config.get('tooltip_account_name_color'),
|
account_name = GLib.markup_escape_text(self.prim_contact.account.name)
|
||||||
GLib.markup_escape_text(prim_contact.account.name))
|
name_markup += " <span foreground='{}'>({})</span>".format(
|
||||||
|
color, account_name)
|
||||||
|
|
||||||
if self.account and helpers.jid_is_blocked(self.account,
|
if account and helpers.jid_is_blocked(account, self.prim_contact.jid):
|
||||||
prim_contact.jid):
|
|
||||||
name_markup += _(' [blocked]')
|
name_markup += _(' [blocked]')
|
||||||
if self.account and \
|
|
||||||
self.account in gajim.interface.minimized_controls and \
|
|
||||||
prim_contact.jid in gajim.interface.minimized_controls[self.account]:
|
|
||||||
name_markup += _(' [minimized]')
|
|
||||||
properties.append((name_markup, None))
|
|
||||||
|
|
||||||
num_resources = 0
|
try:
|
||||||
|
if self.prim_contact.jid in gajim.interface.minimized_controls[account]:
|
||||||
|
name_markup += _(' [minimized]')
|
||||||
|
except KeyError:
|
||||||
|
pass
|
||||||
|
|
||||||
|
self.name.set_markup(name_markup)
|
||||||
|
self.name.show()
|
||||||
|
|
||||||
|
self.num_resources = 0
|
||||||
# put contacts in dict, where key is priority
|
# put contacts in dict, where key is priority
|
||||||
contacts_dict = {}
|
contacts_dict = {}
|
||||||
for contact in contacts:
|
for contact in contacts:
|
||||||
if contact.resource:
|
if contact.resource:
|
||||||
num_resources += 1
|
self.num_resources += 1
|
||||||
if contact.priority in contacts_dict:
|
priority = int(contact.priority)
|
||||||
contacts_dict[int(contact.priority)].append(contact)
|
if priority in contacts_dict:
|
||||||
|
contacts_dict[priority].append(contact)
|
||||||
else:
|
else:
|
||||||
contacts_dict[int(contact.priority)] = [contact]
|
contacts_dict[priority] = [contact]
|
||||||
|
if self.num_resources > 1:
|
||||||
if num_resources > 1:
|
self.status_label.show()
|
||||||
properties.append((_('Status: '), ' '))
|
transport = gajim.get_transport_name_from_jid(self.prim_contact.jid)
|
||||||
transport = gajim.get_transport_name_from_jid(prim_contact.jid)
|
|
||||||
if transport:
|
if transport:
|
||||||
file_path = os.path.join(helpers.get_transport_path(transport),
|
file_path = os.path.join(helpers.get_transport_path(transport),
|
||||||
'16x16')
|
'16x16')
|
||||||
|
@ -505,98 +602,140 @@ class RosterTooltip(NotificationAreaTooltip):
|
||||||
contact_keys.reverse()
|
contact_keys.reverse()
|
||||||
for priority in contact_keys:
|
for priority in contact_keys:
|
||||||
for acontact in contacts_dict[priority]:
|
for acontact in contacts_dict[priority]:
|
||||||
status_line = self.get_status_info(acontact.resource,
|
|
||||||
acontact.priority, acontact.show, acontact.status)
|
|
||||||
|
|
||||||
icon_name = self._get_icon_name_for_tooltip(acontact)
|
icon_name = self._get_icon_name_for_tooltip(acontact)
|
||||||
|
if acontact.status and len(acontact.status) > 25:
|
||||||
|
status = ''
|
||||||
|
add_text = True
|
||||||
|
else:
|
||||||
|
status = acontact.status
|
||||||
|
add_text = False
|
||||||
|
|
||||||
|
status_line = self.get_status_info(acontact.resource,
|
||||||
|
acontact.priority, acontact.show, status)
|
||||||
self.add_status_row(file_path, icon_name, status_line,
|
self.add_status_row(file_path, icon_name, status_line,
|
||||||
acontact.last_status_time)
|
acontact.last_status_time)
|
||||||
properties.append((self.table, None))
|
if add_text:
|
||||||
|
self.add_text_row(acontact.status, 2)
|
||||||
|
|
||||||
else: # only one resource
|
self.tooltip_grid.attach(self.table, 0, 3, 2, 1)
|
||||||
|
self.table.show_all()
|
||||||
|
|
||||||
|
else: # only one resource
|
||||||
if contact.show:
|
if contact.show:
|
||||||
show = helpers.get_uf_show(contact.show)
|
request_time = False
|
||||||
if not self.check_last_time and self.account:
|
try:
|
||||||
|
last_time = self.check_last_time[contact]
|
||||||
|
if isinstance(last_time, float) and last_time < time.time() - 60:
|
||||||
|
request_time = True
|
||||||
|
except KeyError:
|
||||||
|
request_time = True
|
||||||
|
|
||||||
|
if request_time:
|
||||||
if contact.show == 'offline':
|
if contact.show == 'offline':
|
||||||
if not contact.last_status_time:
|
gajim.connections[account].\
|
||||||
gajim.connections[self.account].\
|
request_last_status_time(contact.jid, '')
|
||||||
request_last_status_time(contact.jid, '')
|
|
||||||
else:
|
|
||||||
self.check_last_time = contact.last_status_time
|
|
||||||
elif contact.resource:
|
elif contact.resource:
|
||||||
gajim.connections[self.account].\
|
gajim.connections[account].\
|
||||||
request_last_status_time(
|
request_last_status_time(
|
||||||
contact.jid, contact.resource)
|
contact.jid, contact.resource)
|
||||||
if contact.last_activity_time:
|
self.check_last_time[contact] = time.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')
|
|
||||||
|
|
||||||
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)
|
|
||||||
|
|
||||||
text = text % local_time
|
|
||||||
show += text
|
|
||||||
if self.account and \
|
|
||||||
prim_contact.jid in gajim.gc_connected[self.account]:
|
|
||||||
if gajim.gc_connected[self.account][prim_contact.jid]:
|
|
||||||
show = _('Connected')
|
|
||||||
else:
|
|
||||||
show = _('Disconnected')
|
|
||||||
show = colorize_status(show)
|
|
||||||
|
|
||||||
if contact.status:
|
if contact.status:
|
||||||
status = contact.status.strip()
|
status = contact.status.strip()
|
||||||
if status:
|
if status:
|
||||||
# reduce long status
|
self.status.set_text(status)
|
||||||
# (no more than 300 chars on line and no more than
|
self.status.show()
|
||||||
# 5 lines)
|
self.status_label.show()
|
||||||
# status is wrapped
|
|
||||||
status = helpers.reduce_chars_newlines(status, 300, 5)
|
|
||||||
# escape markup entities.
|
|
||||||
status = GLib.markup_escape_text(status)
|
|
||||||
properties.append(('<i>%s</i>' % status, None))
|
|
||||||
properties.append((show, None))
|
|
||||||
|
|
||||||
self._append_pep_info(contact, properties)
|
# PEP Info
|
||||||
|
self._append_pep_info(contact)
|
||||||
|
|
||||||
properties.append((_('Jabber ID: '), '\u200E' + "<b>%s</b>" % \
|
# JID
|
||||||
prim_contact.jid))
|
self.jid.set_text(self.prim_contact.jid)
|
||||||
|
self.jid.show()
|
||||||
|
self.jid_label.show()
|
||||||
|
|
||||||
# contact has only one ressource
|
# contact has only one ressource
|
||||||
if num_resources == 1 and contact.resource:
|
if self.num_resources == 1 and contact.resource:
|
||||||
properties.append((_('Resource: '), GLib.markup_escape_text(
|
res = GLib.markup_escape_text(contact.resource)
|
||||||
contact.resource) + ' (' + str(contact.priority) + ')'))
|
prio = str(contact.priority)
|
||||||
|
self.resource.set_text("{} ({})".format(res, prio))
|
||||||
|
self.resource.show()
|
||||||
|
self.resource_label.show()
|
||||||
|
|
||||||
if self.account and prim_contact.sub and prim_contact.sub != 'both' and\
|
if self.prim_contact.jid not in gajim.gc_connected[account]:
|
||||||
prim_contact.jid not in gajim.gc_connected[self.account]:
|
if (account and
|
||||||
# ('both' is the normal sub so we don't show it)
|
self.prim_contact.sub and
|
||||||
properties.append(( _('Subscription: '), GLib.markup_escape_text(
|
self.prim_contact.sub != 'both'):
|
||||||
helpers.get_uf_sub(prim_contact.sub))))
|
# ('both' is the normal sub so we don't show it)
|
||||||
|
self.sub.set_text(helpers.get_uf_sub(self.prim_contact.sub))
|
||||||
|
self.sub.show()
|
||||||
|
self.sub_label.show()
|
||||||
|
|
||||||
if prim_contact.keyID:
|
if self.prim_contact.keyID:
|
||||||
keyID = None
|
keyID = None
|
||||||
if len(prim_contact.keyID) == 8:
|
if len(self.prim_contact.keyID) == 8:
|
||||||
keyID = prim_contact.keyID
|
keyID = self.prim_contact.keyID
|
||||||
elif len(prim_contact.keyID) == 16:
|
elif len(self.prim_contact.keyID) == 16:
|
||||||
keyID = prim_contact.keyID[8:]
|
keyID = self.prim_contact.keyID[8:]
|
||||||
if keyID:
|
if keyID:
|
||||||
properties.append((_('OpenPGP: '), GLib.markup_escape_text(
|
self.pgp.set_text(keyID)
|
||||||
keyID)))
|
self.pgp.show()
|
||||||
|
self.pgp_label.show()
|
||||||
|
|
||||||
|
self._set_idle_time(contact)
|
||||||
|
|
||||||
|
# Avatar
|
||||||
|
puny_jid = helpers.sanitize_filename(self.prim_contact.jid)
|
||||||
|
file_ = helpers.get_avatar_path(os.path.join(gajim.AVATAR_PATH,
|
||||||
|
puny_jid))
|
||||||
|
if file_:
|
||||||
|
with open(file_, 'rb') as file_data:
|
||||||
|
pix = gtkgui_helpers.get_pixbuf_from_data(file_data.read())
|
||||||
|
pix = gtkgui_helpers.get_scaled_pixbuf(pix, 'tooltip')
|
||||||
|
self.avatar.set_from_pixbuf(pix)
|
||||||
|
self.avatar.show()
|
||||||
|
|
||||||
|
# Sets the Widget that is at the bottom to expand.
|
||||||
|
# This is needed in case the Picture takes more Space then the Labels
|
||||||
|
i = 1
|
||||||
|
while i < 15:
|
||||||
|
if self.tooltip_grid.get_child_at(0, i):
|
||||||
|
if self.tooltip_grid.get_child_at(0, i).get_visible():
|
||||||
|
self.last_widget = self.tooltip_grid.get_child_at(0, i)
|
||||||
|
i += 1
|
||||||
|
self.last_widget.set_vexpand(True)
|
||||||
|
|
||||||
|
def _append_pep_info(self, contact):
|
||||||
|
"""
|
||||||
|
Append Tune, Mood, Activity, Location information of the specified contact
|
||||||
|
to the given property list.
|
||||||
|
"""
|
||||||
|
if 'mood' in contact.pep:
|
||||||
|
mood = contact.pep['mood'].asMarkupText()
|
||||||
|
self.mood.set_markup(mood)
|
||||||
|
self.mood.show()
|
||||||
|
self.mood_label.show()
|
||||||
|
|
||||||
|
if 'activity' in contact.pep:
|
||||||
|
activity = contact.pep['activity'].asMarkupText()
|
||||||
|
self.activity.set_markup(activity)
|
||||||
|
self.activity.show()
|
||||||
|
self.activity_label.show()
|
||||||
|
|
||||||
|
if 'tune' in contact.pep:
|
||||||
|
tune = contact.pep['tune'].asMarkupText()
|
||||||
|
self.tune.set_markup(tune)
|
||||||
|
self.tune.show()
|
||||||
|
self.tune_label.show()
|
||||||
|
|
||||||
|
if 'location' in contact.pep:
|
||||||
|
location = contact.pep['location'].asMarkupText()
|
||||||
|
self.location.set_markup(location)
|
||||||
|
self.location.show()
|
||||||
|
self.location_label.show()
|
||||||
|
|
||||||
|
def _set_idle_time(self, contact):
|
||||||
if contact.last_activity_time:
|
if contact.last_activity_time:
|
||||||
last_active = datetime(*contact.last_activity_time[:6])
|
last_active = datetime(*contact.last_activity_time[:6])
|
||||||
current = datetime.now()
|
current = datetime.now()
|
||||||
|
@ -613,78 +752,67 @@ class RosterTooltip(NotificationAreaTooltip):
|
||||||
# is no meaningful difference between last activity time and
|
# is no meaningful difference between last activity time and
|
||||||
# current time.
|
# current time.
|
||||||
if diff.days > 0 or diff.seconds > 0:
|
if diff.days > 0 or diff.seconds > 0:
|
||||||
cs = "<span foreground='%s'>" % gajim.config.get(
|
idle_color = gajim.config.get('tooltip_idle_color')
|
||||||
'tooltip_idle_color')
|
idle_markup = "<span foreground='{}'>{}</span>".format(idle_color, formatted)
|
||||||
cs += '%s</span>'
|
self.idle_since.set_markup(idle_markup)
|
||||||
properties.append((str(), None))
|
self.idle_since.show()
|
||||||
idle_since = cs % _("Idle since %s")
|
self.idle_since_label.show()
|
||||||
properties.append((idle_since % formatted, None))
|
idle_markup = "<span foreground='{}'>{}</span>".format(idle_color, str(diff))
|
||||||
idle_for = cs % _("Idle for %s")
|
self.idle_for.set_markup(idle_markup)
|
||||||
properties.append((idle_for % str(diff), None))
|
self.idle_for_label.show()
|
||||||
|
self.idle_for.show()
|
||||||
|
|
||||||
while properties:
|
if contact.show and self.num_resources < 2:
|
||||||
property_ = properties.pop(0)
|
show = helpers.get_uf_show(contact.show)
|
||||||
vcard_current_row += 1
|
if contact.last_status_time:
|
||||||
label = Gtk.Label()
|
if contact.show == 'offline':
|
||||||
if not properties and table_size == 4:
|
text = ' - ' + _('Last status: %s')
|
||||||
label.set_vexpand(True)
|
|
||||||
label.set_halign(Gtk.Align.START)
|
|
||||||
label.set_valign(Gtk.Align.START)
|
|
||||||
if property_[1]:
|
|
||||||
label.set_markup(property_[0])
|
|
||||||
vcard_table.attach(label, 1, vcard_current_row, 1, 1)
|
|
||||||
label = Gtk.Label()
|
|
||||||
if not properties and table_size == 4:
|
|
||||||
label.set_vexpand(True)
|
|
||||||
label.set_halign(Gtk.Align.START)
|
|
||||||
label.set_valign(Gtk.Align.START)
|
|
||||||
label.set_markup(property_[1])
|
|
||||||
label.set_line_wrap(True)
|
|
||||||
vcard_table.attach(label, 2, vcard_current_row, 1, 1)
|
|
||||||
else:
|
|
||||||
if isinstance(property_[0], str):
|
|
||||||
label.set_markup(property_[0])
|
|
||||||
label.set_line_wrap(True)
|
|
||||||
else:
|
else:
|
||||||
label = property_[0]
|
text = _(' since %s')
|
||||||
vcard_table.attach(label, 1, vcard_current_row, 2, 1)
|
|
||||||
self.avatar_image.set_halign(Gtk.Align.START)
|
|
||||||
self.avatar_image.set_valign(Gtk.Align.START)
|
|
||||||
if table_size == 4:
|
|
||||||
vcard_table.attach(self.avatar_image, 3, 2, 1, vcard_current_row - 1)
|
|
||||||
|
|
||||||
gajim.plugin_manager.gui_extension_point('roster_tooltip_populate',
|
if time.strftime('%j', time.localtime()) == \
|
||||||
self, contacts, vcard_table)
|
time.strftime('%j', contact.last_status_time):
|
||||||
self.win.add(vcard_table)
|
# 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)
|
||||||
|
|
||||||
def update_last_time(self, last_time):
|
text = text % local_time
|
||||||
if not self.check_last_time or time.strftime('%x %I:%M %p', last_time) !=\
|
show += text
|
||||||
time.strftime('%x %I:%M %p', self.check_last_time):
|
|
||||||
self.win.destroy()
|
|
||||||
self.win = None
|
|
||||||
self.populate(self.cur_data)
|
|
||||||
self.win.show_all()
|
|
||||||
|
|
||||||
def _append_pep_info(self, contact, properties):
|
# Contact is Groupchat
|
||||||
|
if (self.account and
|
||||||
|
self.prim_contact.jid in gajim.gc_connected[self.account]):
|
||||||
|
if gajim.gc_connected[self.account][self.prim_contact.jid]:
|
||||||
|
show = _('Connected')
|
||||||
|
else:
|
||||||
|
show = _('Disconnected')
|
||||||
|
|
||||||
|
self.user_show.set_markup(colorize_status(show))
|
||||||
|
self.user_show.show()
|
||||||
|
|
||||||
|
def _get_icon_name_for_tooltip(self, contact):
|
||||||
"""
|
"""
|
||||||
Append Tune, Mood, Activity, Location information of the specified contact
|
Helper function used for tooltip contacts/acounts
|
||||||
to the given property list.
|
|
||||||
|
Tooltip on account has fake contact with sub == '', in this case we show
|
||||||
|
real status of the account
|
||||||
"""
|
"""
|
||||||
if 'mood' in contact.pep:
|
if contact.ask == 'subscribe':
|
||||||
mood = contact.pep['mood'].asMarkupText()
|
return 'requested'
|
||||||
properties.append((_('Mood: '), "%s" % mood, None))
|
elif contact.sub in ('both', 'to', ''):
|
||||||
|
return contact.show
|
||||||
|
return 'not in roster'
|
||||||
|
|
||||||
if 'activity' in contact.pep:
|
def update_last_time(self, contact, error=False):
|
||||||
activity = contact.pep['activity'].asMarkupText()
|
if not contact:
|
||||||
properties.append((_('Activity: '), "%s" % activity, None))
|
return
|
||||||
|
if error:
|
||||||
if 'tune' in contact.pep:
|
self.check_last_time[contact] = 'error'
|
||||||
tune = contact.pep['tune'].asMarkupText()
|
return
|
||||||
properties.append((_('Tune: '), "%s" % tune, None))
|
if contact.jid == self.contact_jid:
|
||||||
|
self._set_idle_time(contact)
|
||||||
if 'location' in contact.pep:
|
|
||||||
location = contact.pep['location'].asMarkupText()
|
|
||||||
properties.append((_('Location: '), "%s" % location, None))
|
|
||||||
|
|
||||||
|
|
||||||
class FileTransfersTooltip(BaseTooltip):
|
class FileTransfersTooltip(BaseTooltip):
|
||||||
|
|
Loading…
Add table
Reference in a new issue