2006-07-06 15:59:48 +02:00
|
|
|
import sys
|
2006-05-12 08:34:20 +02:00
|
|
|
|
|
|
|
import dbus
|
|
|
|
import dbus.service
|
|
|
|
import dbus.glib
|
2006-07-15 12:31:06 +02:00
|
|
|
import gtk
|
|
|
|
import gobject
|
2006-05-12 08:34:20 +02:00
|
|
|
|
2006-07-26 12:57:54 +02:00
|
|
|
from sugar.presence.PresenceService import PresenceService
|
|
|
|
|
2006-07-26 01:14:31 +02:00
|
|
|
# Work around for dbus mutex locking issue
|
|
|
|
gtk.gdk.threads_init()
|
|
|
|
dbus.glib.threads_init()
|
|
|
|
|
2006-07-06 16:06:07 +02:00
|
|
|
from sugar.LogWriter import LogWriter
|
2006-07-09 17:37:54 +02:00
|
|
|
import sugar.util
|
2006-07-06 16:06:07 +02:00
|
|
|
|
2006-06-02 20:52:20 +02:00
|
|
|
ACTIVITY_SERVICE_NAME = "com.redhat.Sugar.Activity"
|
|
|
|
ACTIVITY_SERVICE_PATH = "/com/redhat/Sugar/Activity"
|
|
|
|
|
2006-07-26 00:17:05 +02:00
|
|
|
ON_SHARE_CB = "share"
|
2006-06-05 18:35:00 +02:00
|
|
|
|
2006-07-06 15:59:48 +02:00
|
|
|
def get_path(activity_name):
|
|
|
|
"""Returns the activity path"""
|
|
|
|
return '/' + activity_name.replace('.', '/')
|
|
|
|
|
|
|
|
def get_factory(activity_name):
|
|
|
|
"""Returns the activity factory"""
|
|
|
|
return activity_name + '.Factory'
|
|
|
|
|
|
|
|
class ActivityFactory(dbus.service.Object):
|
|
|
|
"""Dbus service that takes care of creating new instances of an activity"""
|
|
|
|
|
2006-07-14 16:40:45 +02:00
|
|
|
def __init__(self, name, activity_class, default_type):
|
|
|
|
self._default_type = default_type
|
2006-07-28 01:25:08 +02:00
|
|
|
|
2006-07-06 15:59:48 +02:00
|
|
|
splitted_module = activity_class.rsplit('.', 1)
|
|
|
|
module_name = splitted_module[0]
|
|
|
|
class_name = splitted_module[1]
|
2006-07-06 23:08:35 +02:00
|
|
|
|
2006-07-28 01:25:08 +02:00
|
|
|
module = __import__(module_name)
|
|
|
|
for comp in module_name.split('.')[1:]:
|
|
|
|
module = getattr(module, comp)
|
2006-07-06 23:08:35 +02:00
|
|
|
|
2006-07-06 15:59:48 +02:00
|
|
|
self._class = getattr(module, class_name)
|
|
|
|
|
|
|
|
bus = dbus.SessionBus()
|
2006-07-14 16:40:45 +02:00
|
|
|
factory = get_factory(name)
|
2006-07-06 15:59:48 +02:00
|
|
|
bus_name = dbus.service.BusName(factory, bus = bus)
|
|
|
|
dbus.service.Object.__init__(self, bus_name, get_path(factory))
|
|
|
|
|
|
|
|
@dbus.service.method("com.redhat.Sugar.ActivityFactory")
|
2006-07-26 12:57:54 +02:00
|
|
|
def create_with_service(self, service_path):
|
|
|
|
pservice = PresenceService()
|
|
|
|
service = pservice._new_object(service_path)
|
|
|
|
activity = self._class(service, [])
|
2006-07-06 15:59:48 +02:00
|
|
|
|
|
|
|
@dbus.service.method("com.redhat.Sugar.ActivityFactory")
|
2006-07-08 15:47:51 +02:00
|
|
|
def create(self):
|
2006-07-14 16:40:45 +02:00
|
|
|
activity = self._class(None, [])
|
|
|
|
activity.set_default_type(self._default_type)
|
2006-07-06 15:59:48 +02:00
|
|
|
|
|
|
|
def create(activity_name, service = None, args = None):
|
|
|
|
"""Create a new activity from his name."""
|
|
|
|
bus = dbus.SessionBus()
|
|
|
|
|
|
|
|
factory_name = get_factory(activity_name)
|
|
|
|
factory_path = get_path(factory_name)
|
|
|
|
|
|
|
|
proxy_obj = bus.get_object(factory_name, factory_path)
|
|
|
|
factory = dbus.Interface(proxy_obj, "com.redhat.Sugar.ActivityFactory")
|
|
|
|
|
2006-07-26 12:57:54 +02:00
|
|
|
if service:
|
|
|
|
print service.object_path()
|
|
|
|
factory.create_with_service(service.object_path())
|
2006-07-06 15:59:48 +02:00
|
|
|
else:
|
2006-07-08 15:47:51 +02:00
|
|
|
factory.create()
|
2006-07-06 15:59:48 +02:00
|
|
|
|
2006-07-14 16:40:45 +02:00
|
|
|
def register_factory(name, activity_class, default_type=None):
|
2006-07-12 22:17:57 +02:00
|
|
|
"""Register the activity factory."""
|
2006-07-14 16:40:45 +02:00
|
|
|
factory = ActivityFactory(name, activity_class, default_type)
|
2006-07-06 15:59:48 +02:00
|
|
|
gtk.main()
|
|
|
|
|
2006-06-02 20:52:20 +02:00
|
|
|
class ActivityDbusService(dbus.service.Object):
|
|
|
|
"""Base dbus service object that each Activity uses to export dbus methods.
|
|
|
|
|
|
|
|
The dbus service is separate from the actual Activity object so that we can
|
2006-07-12 22:17:57 +02:00
|
|
|
tightly control what stuff passes through the dbus python bindings."""
|
2006-06-02 20:52:20 +02:00
|
|
|
|
2006-07-26 00:17:05 +02:00
|
|
|
_ALLOWED_CALLBACKS = [ON_SHARE_CB]
|
2006-06-05 18:35:00 +02:00
|
|
|
|
2006-07-09 17:37:54 +02:00
|
|
|
def __init__(self, xid, activity):
|
2006-06-02 20:52:20 +02:00
|
|
|
self._activity = activity
|
2006-06-05 18:35:00 +02:00
|
|
|
self._callbacks = {}
|
|
|
|
for cb in self._ALLOWED_CALLBACKS:
|
|
|
|
self._callbacks[cb] = None
|
2006-07-09 17:37:54 +02:00
|
|
|
|
|
|
|
bus = dbus.SessionBus()
|
|
|
|
service_name = ACTIVITY_SERVICE_NAME + "%s" % xid
|
|
|
|
object_path = ACTIVITY_SERVICE_PATH + "/%s" % xid
|
|
|
|
service = dbus.service.BusName(service_name, bus=bus)
|
|
|
|
dbus.service.Object.__init__(self, service, object_path)
|
2006-06-02 20:52:20 +02:00
|
|
|
|
2006-06-05 18:35:00 +02:00
|
|
|
def register_callback(self, name, callback):
|
|
|
|
if name not in self._ALLOWED_CALLBACKS:
|
|
|
|
print "ActivityDbusService: bad callback registration request for '%s'" % name
|
|
|
|
return
|
|
|
|
self._callbacks[name] = callback
|
|
|
|
|
2006-06-14 21:06:25 +02:00
|
|
|
def _call_callback_cb(self, func, *args):
|
|
|
|
gobject.idle_add(func, *args)
|
|
|
|
return False
|
|
|
|
|
2006-06-05 18:35:00 +02:00
|
|
|
def _call_callback(self, name, *args):
|
2006-06-14 21:06:25 +02:00
|
|
|
"""Call our activity object back, but from an idle handler
|
|
|
|
to minimize the possibility of stupid activities deadlocking
|
|
|
|
in dbus callbacks."""
|
2006-06-05 18:35:00 +02:00
|
|
|
if name in self._ALLOWED_CALLBACKS and self._callbacks[name]:
|
2006-06-14 21:06:25 +02:00
|
|
|
gobject.idle_add(self._call_callback_cb, self._callbacks[name], *args)
|
2006-06-05 18:35:00 +02:00
|
|
|
|
2006-06-13 20:49:01 +02:00
|
|
|
@dbus.service.method(ACTIVITY_SERVICE_NAME)
|
2006-07-26 00:17:05 +02:00
|
|
|
def share(self):
|
|
|
|
"""Called by the shell to request the activity to share itself on the network."""
|
|
|
|
self._call_callback(ON_SHARE_CB)
|
2006-05-12 08:34:20 +02:00
|
|
|
|
2006-07-09 17:37:54 +02:00
|
|
|
@dbus.service.method(ACTIVITY_SERVICE_NAME)
|
|
|
|
def get_id(self):
|
|
|
|
"""Get the activity identifier"""
|
2006-07-19 19:35:32 +02:00
|
|
|
return self._activity.get_id()
|
2006-07-09 17:37:54 +02:00
|
|
|
|
2006-07-20 12:13:47 +02:00
|
|
|
@dbus.service.method(ACTIVITY_SERVICE_NAME)
|
|
|
|
def get_default_type(self):
|
|
|
|
"""Get the activity default type"""
|
|
|
|
return self._activity.get_default_type()
|
|
|
|
|
2006-07-09 17:37:54 +02:00
|
|
|
@dbus.service.method(ACTIVITY_SERVICE_NAME)
|
|
|
|
def get_shared(self):
|
|
|
|
"""Get the activity identifier"""
|
|
|
|
return self._activity.get_shared()
|
2006-06-23 04:42:29 +02:00
|
|
|
|
2006-07-08 15:47:51 +02:00
|
|
|
class Activity(gtk.Window):
|
2006-06-02 20:52:20 +02:00
|
|
|
"""Base Activity class that all other Activities derive from."""
|
2006-05-12 08:34:20 +02:00
|
|
|
|
2006-07-14 16:40:45 +02:00
|
|
|
def __init__(self, service = None):
|
2006-07-08 15:47:51 +02:00
|
|
|
gtk.Window.__init__(self)
|
|
|
|
|
2006-07-26 12:57:54 +02:00
|
|
|
if service:
|
|
|
|
self._activity_id = service.get_id()
|
2006-07-14 16:40:45 +02:00
|
|
|
self._shared = True
|
2006-07-09 17:37:54 +02:00
|
|
|
else:
|
2006-07-14 16:40:45 +02:00
|
|
|
self._activity_id = sugar.util.unique_id()
|
|
|
|
self._shared = False
|
2006-07-09 17:37:54 +02:00
|
|
|
|
|
|
|
self._dbus_service = None
|
2006-06-18 07:31:55 +02:00
|
|
|
self._initial_service = None
|
2006-06-02 20:52:20 +02:00
|
|
|
self._activity_object = None
|
2006-07-14 16:40:45 +02:00
|
|
|
self._default_type = None
|
2006-06-02 20:52:20 +02:00
|
|
|
|
2006-07-09 17:37:54 +02:00
|
|
|
self.connect('realize', self.__realize)
|
|
|
|
|
2006-07-10 13:42:34 +02:00
|
|
|
self.present()
|
2006-07-09 17:37:54 +02:00
|
|
|
|
|
|
|
def __realize(self, window):
|
2006-07-20 17:52:31 +02:00
|
|
|
group = gtk.Window()
|
|
|
|
group.realize()
|
|
|
|
self.window.set_group(group.window)
|
|
|
|
|
2006-07-09 17:37:54 +02:00
|
|
|
if not self._dbus_service:
|
|
|
|
self._register_service()
|
|
|
|
|
|
|
|
def _register_service(self):
|
|
|
|
self._dbus_service = self._get_new_dbus_service()
|
2006-07-26 00:17:05 +02:00
|
|
|
self._dbus_service.register_callback(ON_SHARE_CB, self._internal_on_share_cb)
|
2006-07-09 17:37:54 +02:00
|
|
|
|
2006-06-02 20:52:20 +02:00
|
|
|
def _cleanup(self):
|
|
|
|
if self._dbus_service:
|
|
|
|
del self._dbus_service
|
|
|
|
self._dbus_service = None
|
|
|
|
|
|
|
|
def __del__(self):
|
|
|
|
self._cleanup()
|
|
|
|
|
|
|
|
def _get_new_dbus_service(self):
|
|
|
|
"""Create and return a new dbus service object for this Activity.
|
|
|
|
Allows subclasses to use their own dbus service object if they choose."""
|
2006-07-09 17:37:54 +02:00
|
|
|
return ActivityDbusService(self.window.xid, self)
|
2006-06-02 20:52:20 +02:00
|
|
|
|
2006-07-14 16:40:45 +02:00
|
|
|
def set_default_type(self, default_type):
|
|
|
|
self._default_type = default_type
|
2006-07-15 12:31:06 +02:00
|
|
|
print self._default_type
|
2006-07-14 16:40:45 +02:00
|
|
|
|
|
|
|
def get_default_type(self):
|
2006-06-17 05:10:30 +02:00
|
|
|
return self._default_type
|
|
|
|
|
2006-07-09 17:37:54 +02:00
|
|
|
def get_shared(self):
|
2006-06-23 04:42:29 +02:00
|
|
|
return self._shared
|
|
|
|
|
2006-06-02 20:52:20 +02:00
|
|
|
def has_focus(self):
|
|
|
|
"""Return whether or not this Activity is visible to the user."""
|
|
|
|
return self._has_focus
|
2006-05-12 08:34:20 +02:00
|
|
|
|
2006-07-26 00:17:05 +02:00
|
|
|
def _internal_on_share_cb(self):
|
2006-07-26 02:04:15 +02:00
|
|
|
"""Callback when the dbus service object tells us the user wishes to share our activity."""
|
|
|
|
if not self._shared:
|
|
|
|
self._shared = True
|
2006-07-26 00:17:05 +02:00
|
|
|
self.share()
|
2006-06-13 20:49:01 +02:00
|
|
|
|
2006-06-15 17:29:00 +02:00
|
|
|
def get_id(self):
|
2006-06-02 20:52:20 +02:00
|
|
|
return self._activity_id
|
2006-05-12 08:34:20 +02:00
|
|
|
|
2006-06-05 18:35:00 +02:00
|
|
|
#############################################################
|
|
|
|
# Pure Virtual methods that subclasses may/may not implement
|
|
|
|
#############################################################
|
|
|
|
|
2006-07-26 00:17:05 +02:00
|
|
|
def share(self):
|
|
|
|
"""Called to request the activity to share itself on the network."""
|
2006-06-13 20:49:01 +02:00
|
|
|
pass
|