2013-10-27 21:59:46 +01:00
# Copyright (C) 2009, 2013 Red Hat, Inc.
2009-07-27 22:30:01 -04:00
# Copyright (C) 2009 Cole Robinson <crobinso@redhat.com>
#
2018-04-04 14:35:41 +01:00
# This work is licensed under the GNU GPLv2 or later.
2018-03-20 15:00:02 -04:00
# See the COPYING file in the top-level directory.
2009-07-27 22:30:01 -04:00
2012-01-31 18:16:54 -05:00
import logging
2018-10-07 15:48:31 -04:00
from gi . repository import Gio
2012-05-14 14:24:56 +01:00
from gi . repository import Gtk
2009-07-27 22:30:01 -04:00
2018-10-07 20:16:55 -04:00
from virtinst import util
2014-09-12 16:10:45 -04:00
from . import vmmenu
from . baseclass import vmmGObject
2018-03-14 13:13:22 -04:00
from . connmanager import vmmConnectionManager
2010-11-23 20:13:50 -05:00
2018-10-07 15:48:31 -04:00
try :
# pylint: disable=ungrouped-imports
from gi . repository import AppIndicator3
except Exception :
AppIndicator3 = None
def _toggle_manager ( * args , * * kwargs ) :
ignore = args
ignore = kwargs
from . manager import vmmManager
manager = vmmManager . get_instance ( None )
if manager . is_visible ( ) :
manager . close ( )
else :
manager . show ( )
2018-10-07 20:16:55 -04:00
def _conn_connect_cb ( src , uri ) :
connmanager = vmmConnectionManager . get_instance ( )
conn = connmanager . conns [ uri ]
if conn . is_disconnected ( ) :
conn . open ( )
def _conn_disconnect_cb ( src , uri ) :
connmanager = vmmConnectionManager . get_instance ( )
conn = connmanager . conns [ uri ]
if not conn . is_disconnected ( ) :
conn . close ( )
2018-10-07 15:48:31 -04:00
def _has_appindicator_dbus ( ) :
try :
bus = Gio . bus_get_sync ( Gio . BusType . SESSION , None )
dbus = Gio . DBusProxy . new_sync ( bus , 0 , None ,
" org.freedesktop.DBus " ,
" /org/freedesktop/DBus " ,
" org.freedesktop.DBus " , None )
if dbus . NameHasOwner ( " (s) " , " org.kde.StatusNotifierWatcher " ) :
return True
if dbus . NameHasOwner ( " (s) " , " org.freedesktop.StatusNotifierWatcher " ) :
return True
return False
except Exception :
logging . exception ( " Error checking for appindicator dbus " )
return False
###########################
# systray backend classes #
###########################
class _Systray ( object ) :
def is_embedded ( self ) :
raise NotImplementedError ( )
def show ( self ) :
raise NotImplementedError ( )
def hide ( self ) :
raise NotImplementedError ( )
def set_menu ( self , menu ) :
raise NotImplementedError ( )
class _SystrayIndicator ( _Systray ) :
def __init__ ( self ) :
self . _icon = AppIndicator3 . Indicator . new (
" virt-manager " , " virt-manager " ,
AppIndicator3 . IndicatorCategory . APPLICATION_STATUS )
def set_menu ( self , menu ) :
hide_item = Gtk . MenuItem . new_with_mnemonic (
_ ( " _Show Virtual Machine Manager " ) )
hide_item . connect ( " activate " , _toggle_manager )
hide_item . show ( )
menu . insert ( hide_item , len ( menu . get_children ( ) ) - 1 )
self . _icon . set_menu ( menu )
self . _icon . set_secondary_activate_target ( hide_item )
def is_embedded ( self ) :
if not self . _icon . get_property ( " connected " ) :
return False
return ( self . _icon . get_status ( ) !=
AppIndicator3 . IndicatorStatus . PASSIVE )
def show ( self ) :
self . _icon . set_status ( AppIndicator3 . IndicatorStatus . ACTIVE )
def hide ( self ) :
self . _icon . set_status ( AppIndicator3 . IndicatorStatus . PASSIVE )
class _SystrayStatusIcon ( _Systray ) :
def __init__ ( self ) :
self . _icon = Gtk . StatusIcon ( )
self . _icon . set_property ( " icon-name " , " virt-manager " )
self . _icon . connect ( " activate " , _toggle_manager )
self . _icon . connect ( " popup-menu " , self . _popup_cb )
self . _icon . set_tooltip_text ( _ ( " Virtual Machine Manager " ) )
self . _menu = None
def is_embedded ( self ) :
return self . _icon . is_embedded ( )
def set_menu ( self , menu ) :
self . _menu = menu
def _popup_cb ( self , src , button , event_time ) :
if button != 3 :
return
self . _menu . popup ( None , None ,
Gtk . StatusIcon . position_menu ,
self . _icon , 0 , event_time )
def show ( self ) :
self . _icon . set_visible ( True )
def hide ( self ) :
self . _icon . set_visible ( False )
2013-04-13 14:34:52 -04:00
2010-12-09 14:06:00 -05:00
class vmmSystray ( vmmGObject ) :
2018-03-15 05:53:58 -04:00
@classmethod
def get_instance ( cls ) :
if not cls . _instance :
2018-03-17 16:08:20 -04:00
cls . _instance = vmmSystray ( )
2018-03-15 05:53:58 -04:00
return cls . _instance
2012-05-14 14:24:56 +01:00
2018-03-14 13:13:22 -04:00
def __init__ ( self ) :
2010-12-09 14:06:00 -05:00
vmmGObject . __init__ ( self )
2018-03-15 07:43:56 -04:00
self . _cleanup_on_app_close ( )
2018-10-07 15:48:31 -04:00
self . topwin = None # Need this for error callbacks from VMActionMenu
2009-07-27 22:30:01 -04:00
2018-10-07 15:48:31 -04:00
self . _systray = None
self . _using_appindicator = False
2009-07-27 22:30:01 -04:00
2018-10-07 15:48:31 -04:00
if AppIndicator3 :
if not _has_appindicator_dbus ( ) :
logging . debug ( " AppIndicator3 is available, but didn ' t "
" find any dbus watcher. " )
else :
self . _using_appindicator = True
logging . debug ( " Using AppIndicator3 for systray " )
2018-03-15 05:53:58 -04:00
2018-10-07 15:48:31 -04:00
connmanager = vmmConnectionManager . get_instance ( )
connmanager . connect ( " conn-added " , self . _conn_added_cb )
connmanager . connect ( " conn-removed " , self . _rebuild_menu )
for conn in connmanager . conns . values ( ) :
self . _conn_added_cb ( connmanager , conn )
2009-07-27 22:30:01 -04:00
2014-09-28 13:37:16 +02:00
self . add_gsettings_handle (
2018-03-15 05:53:58 -04:00
self . config . on_view_system_tray_changed (
self . _show_systray_changed_cb ) )
2018-10-07 15:48:31 -04:00
self . _startup ( )
2009-07-27 22:30:01 -04:00
2018-03-14 10:29:20 -04:00
2018-03-17 11:46:38 -04:00
def is_embedded ( self ) :
2018-10-07 15:48:31 -04:00
return self . _systray and self . _systray . is_embedded ( )
2009-11-10 11:56:15 -05:00
2011-07-23 21:16:54 -04:00
def _cleanup ( self ) :
2018-10-07 15:48:31 -04:00
self . _hide_systray ( )
self . _systray = None
2015-09-20 14:20:50 -04:00
2011-04-13 09:27:02 -04:00
2015-09-20 14:20:50 -04:00
###########################
# Initialization routines #
###########################
2009-07-27 22:30:01 -04:00
2018-10-07 15:48:31 -04:00
def _show_systray ( self ) :
if not self . _systray :
if self . _using_appindicator :
self . _systray = _SystrayIndicator ( )
else :
self . _systray = _SystrayStatusIcon ( )
self . _rebuild_menu ( force = True )
self . _systray . show ( )
def _hide_systray ( self ) :
if not self . _systray :
2018-03-17 11:37:58 -04:00
return
2018-10-07 15:48:31 -04:00
self . _systray . hide ( )
2009-07-27 22:30:01 -04:00
2018-03-15 05:53:58 -04:00
def _show_systray_changed_cb ( self ) :
2009-07-27 22:30:01 -04:00
do_show = self . config . get_view_system_tray ( )
2012-01-31 18:16:54 -05:00
logging . debug ( " Showing systray: %s " , do_show )
2009-07-27 22:30:01 -04:00
2018-03-15 05:53:58 -04:00
if do_show :
2018-10-07 15:48:31 -04:00
self . _show_systray ( )
2018-03-15 05:53:58 -04:00
else :
2018-10-07 15:48:31 -04:00
self . _hide_systray ( )
2009-07-27 22:30:01 -04:00
2018-10-07 15:48:31 -04:00
def _startup ( self ) :
# This will trigger the actual UI showing
self . _show_systray_changed_cb ( )
2010-03-24 10:58:17 -04:00
2018-10-07 15:48:31 -04:00
#################
# Menu building #
#################
2009-07-27 22:30:01 -04:00
2018-10-07 15:48:31 -04:00
def _build_vm_menuitem ( self , vm ) :
menu_item = Gtk . ImageMenuItem . new_with_label ( vm . get_name_or_title ( ) )
menu_item . set_use_underline ( False )
vm_action_menu = vmmenu . VMActionMenu ( self , lambda : vm )
vm_action_menu . update_widget_states ( vm )
menu_item . set_submenu ( vm_action_menu )
return menu_item
2009-07-27 22:30:01 -04:00
2018-10-07 15:48:31 -04:00
def _build_conn_menuitem ( self , conn ) :
2014-07-04 18:20:08 -04:00
menu_item = Gtk . MenuItem . new_with_label ( conn . get_pretty_desc ( ) )
2018-10-07 20:16:55 -04:00
if conn . is_active ( ) :
label = menu_item . get_child ( )
markup = " <b> %s </b> " % util . xml_escape ( conn . get_pretty_desc ( ) )
label . set_markup ( markup )
2009-07-27 22:30:01 -04:00
2018-10-07 15:48:31 -04:00
menu = Gtk . Menu ( )
vms = conn . list_vms ( )
vms . sort ( key = lambda v : v . get_name_or_title ( ) )
2009-07-27 22:30:01 -04:00
2018-10-07 15:48:31 -04:00
for vm in vms :
menu . add ( self . _build_vm_menuitem ( vm ) )
if not vms :
vmitem = Gtk . MenuItem . new_with_label ( _ ( " No virtual machines " ) )
vmitem . set_sensitive ( False )
menu . add ( vmitem )
2009-07-27 22:30:01 -04:00
2018-10-07 20:16:55 -04:00
menu . add ( Gtk . SeparatorMenuItem ( ) )
if conn . is_active ( ) :
citem = Gtk . ImageMenuItem . new_from_stock (
Gtk . STOCK_DISCONNECT , None )
citem . connect ( " activate " , _conn_disconnect_cb , conn . get_uri ( ) )
else :
citem = Gtk . ImageMenuItem . new_from_stock ( Gtk . STOCK_CONNECT , None )
citem . connect ( " activate " , _conn_connect_cb , conn . get_uri ( ) )
menu . add ( citem )
2018-10-07 15:48:31 -04:00
menu_item . set_submenu ( menu )
return menu_item
2009-07-27 22:30:01 -04:00
2018-10-07 15:48:31 -04:00
def _build_menu ( self ) :
connmanager = vmmConnectionManager . get_instance ( )
2018-10-07 20:16:55 -04:00
conns = list ( connmanager . conns . values ( ) )
2018-10-07 15:48:31 -04:00
menu = Gtk . Menu ( )
2009-07-27 22:30:01 -04:00
2018-10-07 20:16:55 -04:00
conns . sort ( key = lambda c : c . get_pretty_desc ( ) . lower ( ) )
conns . sort ( key = lambda c : not c . is_active ( ) )
for conn in conns :
connmenu = self . _build_conn_menuitem ( conn )
2018-10-07 15:48:31 -04:00
menu . add ( connmenu )
2009-07-27 22:30:01 -04:00
2018-10-07 15:48:31 -04:00
menu . add ( Gtk . SeparatorMenuItem ( ) )
2009-07-27 22:30:01 -04:00
2018-10-07 15:48:31 -04:00
exit_item = Gtk . ImageMenuItem . new_from_stock ( Gtk . STOCK_QUIT , None )
exit_item . connect ( " activate " , self . _exit_app_cb )
menu . add ( exit_item )
menu . show_all ( )
return menu
def _rebuild_menu ( self , * args , * * kwargs ) :
ignore = args
ignore = kwargs
if " force " not in kwargs and not self . is_embedded ( ) :
2014-06-02 17:17:47 -04:00
return
2018-10-07 15:48:31 -04:00
if vmmConnectionManager . get_instance ( ) is True :
# In app cleanup, don't do anything
2009-07-27 22:30:01 -04:00
return
2018-10-07 15:48:31 -04:00
# Yeah, this is kinda nutty, we rebuild the whole menu widget
# on any conn or VM state change. We kinda need to do this
# for appindicator, because we communicate with the remote
# UI via changing the menu widget, unlike statusicon which
# we could delay until 'show' time. This is likely slow as
# dirt for a virt-manager instance with a lot of connections
# and VMs...
menu = self . _build_menu ( )
self . _systray . set_menu ( menu )
def _conn_added_cb ( self , src , conn ) :
conn . connect ( " vm-added " , self . _vm_added_cb )
conn . connect ( " vm-removed " , self . _rebuild_menu )
conn . connect ( " state-changed " , self . _rebuild_menu )
self . _rebuild_menu ( )
def _vm_added_cb ( self , conn , connkey ) :
vm = conn . get_vm ( connkey )
vm . connect ( " state-changed " , self . _rebuild_menu )
self . _rebuild_menu ( )
2009-07-27 22:30:01 -04:00
2018-10-07 15:48:31 -04:00
def _exit_app_cb ( self , src ) :
2018-03-17 11:46:38 -04:00
from . engine import vmmEngine
2018-03-14 18:58:22 -04:00
vmmEngine . get_instance ( ) . exit_app ( )