364 lines
13 KiB
Python
364 lines
13 KiB
Python
|
# Copyright (C) 2006 Nikos Kouremenos <kourem AT gmail.com>
|
||
|
# Copyright (C) 2006-2007 Jean-Marie Traissard <jim AT lapin.org>
|
||
|
# Copyright (C) 2006-2014 Yann Leboulanger <asterix AT lagaule.org>
|
||
|
# Copyright (C) 2007 Lukas Petrovicky <lukas AT petrovicky.net>
|
||
|
# Julien Pivotto <roidelapluie AT gmail.com>
|
||
|
# Copyright (C) 2008 Jonathan Schleifer <js-gajim AT webkeks.org>
|
||
|
#
|
||
|
# 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
|
||
|
# by the Free Software Foundation; version 3 only.
|
||
|
#
|
||
|
# 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.
|
||
|
#
|
||
|
# You should have received a copy of the GNU General Public License
|
||
|
# along with Gajim. If not, see <http://www.gnu.org/licenses/>.
|
||
|
|
||
|
import os
|
||
|
|
||
|
from gi.repository import Gtk
|
||
|
from gi.repository import GLib
|
||
|
|
||
|
from gajim.common import app
|
||
|
from gajim.common import helpers
|
||
|
from gajim.common.i18n import _
|
||
|
from gajim.common.helpers import save_roster_position
|
||
|
|
||
|
from .util import get_builder
|
||
|
from .util import get_icon_name
|
||
|
from .util import restore_roster_position
|
||
|
from .util import open_window
|
||
|
from .single_message import SingleMessageWindow
|
||
|
from .tooltips import NotificationAreaTooltip
|
||
|
|
||
|
|
||
|
class StatusIcon:
|
||
|
"""
|
||
|
Class for the notification area icon
|
||
|
"""
|
||
|
|
||
|
def __init__(self):
|
||
|
self._single_message_handler_id = None
|
||
|
self._show_roster_handler_id = None
|
||
|
# click somewhere else does not popdown menu. workaround this.
|
||
|
self.added_hide_menuitem = False
|
||
|
self.status = 'offline'
|
||
|
self._ui = get_builder('systray_context_menu.ui')
|
||
|
self.systray_context_menu = self._ui.systray_context_menu
|
||
|
self._ui.connect_signals(self)
|
||
|
self.popup_menus = []
|
||
|
self.status_icon = None
|
||
|
self.tooltip = NotificationAreaTooltip()
|
||
|
self._icon_size = '16'
|
||
|
|
||
|
def show_icon(self):
|
||
|
if not self.status_icon:
|
||
|
self.status_icon = Gtk.StatusIcon()
|
||
|
self.status_icon.set_property('has-tooltip', True)
|
||
|
self.status_icon.connect('activate', self._on_activate)
|
||
|
self.status_icon.connect('popup-menu', self._on_popup_menu)
|
||
|
self.status_icon.connect('query-tooltip', self._on_query_tooltip)
|
||
|
self.status_icon.connect('size-changed', self.set_img)
|
||
|
|
||
|
self.set_img()
|
||
|
self._subscribe_events()
|
||
|
|
||
|
def hide_icon(self):
|
||
|
self.status_icon.set_visible(False)
|
||
|
self._unsubscribe_events()
|
||
|
|
||
|
def change_status(self, global_status):
|
||
|
"""
|
||
|
Set tray image to 'global_status'
|
||
|
"""
|
||
|
# change image and status, only if it is different
|
||
|
if global_status is not None and self.status != global_status:
|
||
|
self.status = global_status
|
||
|
self.set_img()
|
||
|
|
||
|
def _subscribe_events(self):
|
||
|
"""
|
||
|
Register listeners to the events class
|
||
|
"""
|
||
|
app.events.event_added_subscribe(self._on_event_added)
|
||
|
app.events.event_removed_subscribe(self._on_event_removed)
|
||
|
|
||
|
def _unsubscribe_events(self):
|
||
|
"""
|
||
|
Unregister listeners to the events class
|
||
|
"""
|
||
|
app.events.event_added_unsubscribe(self._on_event_added)
|
||
|
app.events.event_removed_unsubscribe(self._on_event_removed)
|
||
|
|
||
|
def _on_event_added(self, event):
|
||
|
"""
|
||
|
Called when an event is added to the event list
|
||
|
"""
|
||
|
if event.show_in_systray:
|
||
|
self.set_img()
|
||
|
|
||
|
def _on_event_removed(self, _event_list):
|
||
|
"""
|
||
|
Called when one or more events are removed from the event list
|
||
|
"""
|
||
|
self.set_img()
|
||
|
|
||
|
def _on_query_tooltip(self, _status_icon, _x, _y, _keyboard_mode, tooltip):
|
||
|
tooltip.set_custom(self.tooltip.get_tooltip())
|
||
|
return True
|
||
|
|
||
|
def _on_popup_menu(self, _status_icon, button, activate_time):
|
||
|
if button == 1: # Left click
|
||
|
self._on_left_click()
|
||
|
elif button == 2: # middle click
|
||
|
self._on_middle_click()
|
||
|
elif button == 3: # right click
|
||
|
self._make_menu(button, activate_time)
|
||
|
|
||
|
def _on_activate(self, _status_icon):
|
||
|
self._on_left_click()
|
||
|
|
||
|
def on_status_icon_size_changed(self, _statusicon, size):
|
||
|
if size > 31:
|
||
|
self._icon_size = '32'
|
||
|
elif size > 23:
|
||
|
self._icon_size = '24'
|
||
|
else:
|
||
|
self._icon_size = '16'
|
||
|
if os.environ.get('KDE_FULL_SESSION') == 'true':
|
||
|
# detect KDE session. see #5476
|
||
|
self._icon_size = '32'
|
||
|
self.set_img()
|
||
|
|
||
|
def set_img(self, *args):
|
||
|
"""
|
||
|
Apart from image, we also update tooltip text here
|
||
|
"""
|
||
|
if not app.interface.systray_enabled:
|
||
|
return
|
||
|
if app.settings.get('trayicon') == 'always':
|
||
|
self.status_icon.set_visible(True)
|
||
|
if app.events.get_nb_systray_events():
|
||
|
self.status_icon.set_visible(True)
|
||
|
|
||
|
icon_name = get_icon_name('event')
|
||
|
self.status_icon.set_from_icon_name(icon_name)
|
||
|
return
|
||
|
|
||
|
if app.settings.get('trayicon') == 'on_event':
|
||
|
self.status_icon.set_visible(False)
|
||
|
|
||
|
icon_name = get_icon_name(self.status)
|
||
|
self.status_icon.set_from_icon_name(icon_name)
|
||
|
|
||
|
@staticmethod
|
||
|
def _on_single_message(_widget, account):
|
||
|
SingleMessageWindow(account, action='send')
|
||
|
|
||
|
@staticmethod
|
||
|
def _on_new_chat(_widget):
|
||
|
app.app.activate_action('start-chat', GLib.Variant('s', ''))
|
||
|
|
||
|
def _make_menu(self, _event_button, event_time):
|
||
|
"""
|
||
|
Create chat with and new message (sub) menus/menuitems
|
||
|
"""
|
||
|
for menu in self.popup_menus:
|
||
|
menu.destroy()
|
||
|
|
||
|
start_chat_menuitem = self._ui.start_chat_menuitem
|
||
|
single_message_menuitem = self._ui.single_message_menuitem
|
||
|
status_menuitem = self._ui.status_menu
|
||
|
sounds_mute_menuitem = self._ui.sounds_mute_menuitem
|
||
|
show_roster_menuitem = self._ui.show_roster_menuitem
|
||
|
|
||
|
if self._single_message_handler_id:
|
||
|
single_message_menuitem.handler_disconnect(
|
||
|
self._single_message_handler_id)
|
||
|
self._single_message_handler_id = None
|
||
|
|
||
|
sub_menu = Gtk.Menu()
|
||
|
self.popup_menus.append(sub_menu)
|
||
|
status_menuitem.set_submenu(sub_menu)
|
||
|
|
||
|
for show in ('online', 'away', 'xa', 'dnd'):
|
||
|
uf_show = helpers.get_uf_show(show, use_mnemonic=True)
|
||
|
item = Gtk.MenuItem.new_with_mnemonic(uf_show)
|
||
|
sub_menu.append(item)
|
||
|
item.connect('activate', self._on_show, show)
|
||
|
|
||
|
item = Gtk.SeparatorMenuItem.new()
|
||
|
sub_menu.append(item)
|
||
|
|
||
|
item = Gtk.MenuItem.new_with_mnemonic(_('_Change Status Message…'))
|
||
|
sub_menu.append(item)
|
||
|
item.connect('activate', self._on_change_status)
|
||
|
|
||
|
connected_accounts = app.get_number_of_connected_accounts()
|
||
|
if connected_accounts < 1:
|
||
|
item.set_sensitive(False)
|
||
|
|
||
|
item = Gtk.SeparatorMenuItem.new()
|
||
|
sub_menu.append(item)
|
||
|
|
||
|
uf_show = helpers.get_uf_show('offline', use_mnemonic=True)
|
||
|
item = Gtk.MenuItem.new_with_mnemonic(uf_show)
|
||
|
sub_menu.append(item)
|
||
|
item.connect('activate', self._on_show, 'offline')
|
||
|
|
||
|
is_zeroconf = connected_accounts == 1 and app.zeroconf_is_connected()
|
||
|
iskey = connected_accounts > 0 and not is_zeroconf
|
||
|
start_chat_menuitem.set_sensitive(iskey)
|
||
|
single_message_menuitem.set_sensitive(iskey)
|
||
|
|
||
|
accounts_list = sorted(app.contacts.get_accounts())
|
||
|
|
||
|
# menu items that don't apply to zeroconf connections
|
||
|
if connected_accounts == 1 or (connected_accounts == 2 and \
|
||
|
app.zeroconf_is_connected()):
|
||
|
# only one 'real' (non-zeroconf) account is connected, don't need
|
||
|
# submenus
|
||
|
for account in app.connections:
|
||
|
if app.account_is_available(account) and \
|
||
|
not app.settings.get_account_setting(account, 'is_zeroconf'):
|
||
|
|
||
|
# for single message
|
||
|
single_message_menuitem.set_submenu(None)
|
||
|
self._single_message_handler_id = single_message_menuitem.\
|
||
|
connect('activate',
|
||
|
self._on_single_message, account)
|
||
|
break # No other account connected
|
||
|
else:
|
||
|
# 2 or more 'real' accounts are connected, make submenus
|
||
|
account_menu_for_single_message = Gtk.Menu()
|
||
|
single_message_menuitem.set_submenu(
|
||
|
account_menu_for_single_message)
|
||
|
self.popup_menus.append(account_menu_for_single_message)
|
||
|
|
||
|
for account in accounts_list:
|
||
|
account_label = app.get_account_label(account)
|
||
|
if app.connections[account].is_zeroconf or \
|
||
|
not app.account_is_available(account):
|
||
|
continue
|
||
|
# for single message
|
||
|
item = Gtk.MenuItem.new_with_label(
|
||
|
_('using account %s') % account_label)
|
||
|
item.connect('activate',
|
||
|
self._on_single_message, account)
|
||
|
account_menu_for_single_message.append(item)
|
||
|
|
||
|
sounds_mute_menuitem.set_active(not app.settings.get('sounds_on'))
|
||
|
|
||
|
win = app.interface.roster.window
|
||
|
if self._show_roster_handler_id:
|
||
|
show_roster_menuitem.handler_disconnect(
|
||
|
self._show_roster_handler_id)
|
||
|
if win.get_property('has-toplevel-focus'):
|
||
|
show_roster_menuitem.get_children()[0].set_label(
|
||
|
_('Hide _Contact List'))
|
||
|
self._show_roster_handler_id = show_roster_menuitem.connect(
|
||
|
'activate', self._on_hide_roster)
|
||
|
else:
|
||
|
show_roster_menuitem.get_children()[0].set_label(
|
||
|
_('Show _Contact List'))
|
||
|
self._show_roster_handler_id = show_roster_menuitem.connect(
|
||
|
'activate', self._on_show_roster)
|
||
|
|
||
|
if os.name == 'nt':
|
||
|
if self.added_hide_menuitem is False:
|
||
|
self.systray_context_menu.prepend(Gtk.SeparatorMenuItem.new())
|
||
|
item = Gtk.MenuItem.new_with_label(
|
||
|
_('Hide this menu'))
|
||
|
self.systray_context_menu.prepend(item)
|
||
|
self.added_hide_menuitem = True
|
||
|
|
||
|
self.systray_context_menu.show_all()
|
||
|
self.systray_context_menu.popup(None, None, None, None, 0, event_time)
|
||
|
|
||
|
@staticmethod
|
||
|
def _on_show_all_events(_widget):
|
||
|
events = app.events.get_systray_events()
|
||
|
for account in events:
|
||
|
for jid in events[account]:
|
||
|
for event in events[account][jid]:
|
||
|
app.interface.handle_event(account, jid, event.type_)
|
||
|
|
||
|
@staticmethod
|
||
|
def _on_sounds_mute(widget):
|
||
|
app.settings.set('sounds_on', not widget.get_active())
|
||
|
|
||
|
@staticmethod
|
||
|
def _on_show_roster(_widget):
|
||
|
win = app.interface.roster.window
|
||
|
win.present()
|
||
|
|
||
|
@staticmethod
|
||
|
def _on_hide_roster(_widget):
|
||
|
win = app.interface.roster.window
|
||
|
win.hide()
|
||
|
|
||
|
@staticmethod
|
||
|
def _on_preferences(_widget):
|
||
|
open_window('Preferences')
|
||
|
|
||
|
@staticmethod
|
||
|
def _on_quit(_widget):
|
||
|
app.interface.roster.on_quit_request()
|
||
|
|
||
|
def _on_left_click(self):
|
||
|
win = app.interface.roster.window
|
||
|
if app.events.get_systray_events():
|
||
|
self._handle_first_event()
|
||
|
return
|
||
|
|
||
|
if win.get_property('has-toplevel-focus'):
|
||
|
save_roster_position(win)
|
||
|
win.hide()
|
||
|
return
|
||
|
|
||
|
visible = win.get_property('visible')
|
||
|
win.show_all()
|
||
|
if not visible:
|
||
|
# Window was minimized
|
||
|
restore_roster_position(win)
|
||
|
|
||
|
if not app.settings.get('roster_window_skip_taskbar'):
|
||
|
win.set_property('skip-taskbar-hint', False)
|
||
|
win.present_with_time(Gtk.get_current_event_time())
|
||
|
|
||
|
@staticmethod
|
||
|
def _handle_first_event():
|
||
|
account, jid, event = app.events.get_first_systray_event()
|
||
|
if not event:
|
||
|
return
|
||
|
win = app.interface.roster.window
|
||
|
if not win.get_property('visible'):
|
||
|
# Needed if we are in one window mode
|
||
|
restore_roster_position(win)
|
||
|
app.interface.handle_event(account, jid, event.type_)
|
||
|
|
||
|
@staticmethod
|
||
|
def _on_middle_click():
|
||
|
"""
|
||
|
Middle click raises window to have complete focus (fe. get kbd events)
|
||
|
but if already raised, it hides it
|
||
|
"""
|
||
|
win = app.interface.roster.window
|
||
|
if win.is_active():
|
||
|
win.hide()
|
||
|
else:
|
||
|
win.present()
|
||
|
|
||
|
@staticmethod
|
||
|
def _on_show(_widget, show):
|
||
|
app.interface.change_status(status=show)
|
||
|
|
||
|
@staticmethod
|
||
|
def _on_change_status(_widget):
|
||
|
app.interface.change_status()
|