184 lines
6.1 KiB
Python
184 lines
6.1 KiB
Python
"""
|
|
nautilus-gsconnect.py - A Nautilus extension for sending files via GSConnect.
|
|
|
|
A great deal of credit and appreciation is owed to the indicator-kdeconnect
|
|
developers for the sister Python script 'kdeconnect-send-nautilus.py':
|
|
|
|
https://github.com/Bajoja/indicator-kdeconnect/blob/master/data/extensions/kdeconnect-send-nautilus.py
|
|
"""
|
|
|
|
import gettext
|
|
import os.path
|
|
import sys
|
|
|
|
import gi
|
|
gi.require_version('Gio', '2.0')
|
|
gi.require_version('GLib', '2.0')
|
|
gi.require_version('GObject', '2.0')
|
|
from gi.repository import Gio, GLib, GObject
|
|
|
|
|
|
# Host application detection
|
|
#
|
|
# Nemo seems to reliably identify itself as 'nemo' in argv[0], so we
|
|
# can test for that. Nautilus detection is less reliable, so don't try.
|
|
# See https://github.com/linuxmint/nemo-extensions/issues/330
|
|
if "nemo" in sys.argv[0].lower():
|
|
# Host runtime is nemo-python
|
|
gi.require_version('Nemo', '3.0')
|
|
from gi.repository import Nemo as FileManager
|
|
else:
|
|
# Otherwise, just assume it's nautilus-python
|
|
gi.require_version('Nautilus', '3.0')
|
|
from gi.repository import Nautilus as FileManager
|
|
|
|
|
|
SERVICE_NAME = 'org.gnome.Shell.Extensions.GSConnect'
|
|
SERVICE_PATH = '/org/gnome/Shell/Extensions/GSConnect'
|
|
|
|
# Init gettext translations
|
|
LOCALE_DIR = os.path.join(GLib.get_user_data_dir(),
|
|
'gnome-shell', 'extensions',
|
|
'gsconnect@andyholmes.github.io', 'locale')
|
|
|
|
if not os.path.exists(LOCALE_DIR):
|
|
LOCALE_DIR = None
|
|
|
|
try:
|
|
i18n = gettext.translation(SERVICE_NAME,
|
|
localedir=LOCALE_DIR)
|
|
_ = i18n.gettext
|
|
|
|
except (IOError, OSError) as e:
|
|
print('GSConnect: {0}'.format(e.strerror))
|
|
i18n = gettext.translation(SERVICE_NAME,
|
|
localedir=LOCALE_DIR,
|
|
fallback=True)
|
|
_ = i18n.gettext
|
|
|
|
|
|
class GSConnectShareExtension(GObject.Object, FileManager.MenuProvider):
|
|
"""A context menu for sending files via GSConnect."""
|
|
|
|
def __init__(self):
|
|
"""Initialize the DBus ObjectManager"""
|
|
|
|
GObject.Object.__init__(self)
|
|
|
|
self.devices = {}
|
|
|
|
Gio.DBusProxy.new_for_bus(Gio.BusType.SESSION,
|
|
Gio.DBusProxyFlags.DO_NOT_AUTO_START,
|
|
None,
|
|
SERVICE_NAME,
|
|
SERVICE_PATH,
|
|
'org.freedesktop.DBus.ObjectManager',
|
|
None,
|
|
self._init_async,
|
|
None)
|
|
|
|
def _init_async(self, proxy, res, user_data):
|
|
proxy = proxy.new_for_bus_finish(res)
|
|
proxy.connect('notify::g-name-owner', self._on_name_owner_changed)
|
|
proxy.connect('g-signal', self._on_g_signal)
|
|
|
|
self._on_name_owner_changed(proxy, None)
|
|
|
|
def _on_g_signal(self, proxy, sender_name, signal_name, parameters):
|
|
# Wait until the service is ready
|
|
if proxy.props.g_name_owner is None:
|
|
return
|
|
|
|
objects = parameters.unpack()
|
|
|
|
if signal_name == 'InterfacesAdded':
|
|
for object_path, props in objects.items():
|
|
props = props['org.gnome.Shell.Extensions.GSConnect.Device']
|
|
|
|
self.devices[object_path] = (props['Name'],
|
|
Gio.DBusActionGroup.get(
|
|
proxy.get_connection(),
|
|
SERVICE_NAME,
|
|
object_path))
|
|
elif signal_name == 'InterfacesRemoved':
|
|
for object_path in objects:
|
|
try:
|
|
del self.devices[object_path]
|
|
except KeyError:
|
|
pass
|
|
|
|
def _on_name_owner_changed(self, proxy, pspec):
|
|
# Wait until the service is ready
|
|
if proxy.props.g_name_owner is None:
|
|
self.devices = {}
|
|
else:
|
|
proxy.call('GetManagedObjects',
|
|
None,
|
|
Gio.DBusCallFlags.NO_AUTO_START,
|
|
-1,
|
|
None,
|
|
self._get_managed_objects,
|
|
None)
|
|
|
|
def _get_managed_objects(self, proxy, res, user_data):
|
|
objects = proxy.call_finish(res)[0]
|
|
|
|
for object_path, props in objects.items():
|
|
props = props['org.gnome.Shell.Extensions.GSConnect.Device']
|
|
|
|
self.devices[object_path] = (props['Name'],
|
|
Gio.DBusActionGroup.get(
|
|
proxy.get_connection(),
|
|
SERVICE_NAME,
|
|
object_path))
|
|
|
|
def send_files(self, menu, files, action_group):
|
|
"""Send *files* to *device_id*"""
|
|
|
|
for file in files:
|
|
variant = GLib.Variant('(sb)', (file.get_uri(), False))
|
|
action_group.activate_action('shareFile', variant)
|
|
|
|
def get_file_items(self, window, files):
|
|
"""Return a list of select files to be sent"""
|
|
|
|
# Only accept regular files
|
|
for uri in files:
|
|
if uri.get_uri_scheme() != 'file' or uri.is_directory():
|
|
return ()
|
|
|
|
# Enumerate capable devices
|
|
devices = []
|
|
|
|
for name, action_group in self.devices.values():
|
|
if action_group.get_action_enabled('shareFile'):
|
|
devices.append([name, action_group])
|
|
|
|
# No capable devices; don't show menu entry
|
|
if not devices:
|
|
return ()
|
|
|
|
# Context Menu Item
|
|
menu = FileManager.MenuItem(
|
|
name='GSConnectShareExtension::Devices',
|
|
label=_('Send To Mobile Device')
|
|
)
|
|
|
|
# Context Submenu
|
|
submenu = FileManager.Menu()
|
|
menu.set_submenu(submenu)
|
|
|
|
# Context Submenu Items
|
|
for name, action_group in devices:
|
|
item = FileManager.MenuItem(
|
|
name='GSConnectShareExtension::Device' + name,
|
|
label=name
|
|
)
|
|
|
|
item.connect('activate', self.send_files, files, action_group)
|
|
|
|
submenu.append_item(item)
|
|
|
|
return (menu,)
|
|
|