# Copyright (C) 2006 Nikos Kouremenos # Copyright (C) 2006-2007 Jean-Marie Traissard # Copyright (C) 2006-2014 Yann Leboulanger # Copyright (C) 2007 Lukas Petrovicky # Julien Pivotto # Copyright (C) 2008 Jonathan Schleifer # # 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 . 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()