2006-10-15 01:24:45 +02:00
|
|
|
# Copyright (C) 2006, Red Hat, Inc.
|
|
|
|
#
|
|
|
|
# This program 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; either version 2 of the License, or
|
|
|
|
# (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This program 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 this program; if not, write to the Free Software
|
|
|
|
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
|
2006-12-14 16:45:52 +01:00
|
|
|
import logging
|
|
|
|
|
2006-07-08 15:47:51 +02:00
|
|
|
import gtk
|
2006-07-12 22:17:57 +02:00
|
|
|
import gobject
|
2006-07-27 10:35:59 +02:00
|
|
|
import wnck
|
2006-12-14 16:45:52 +01:00
|
|
|
import dbus
|
2006-07-06 23:34:23 +02:00
|
|
|
|
2006-10-05 13:26:02 +02:00
|
|
|
import view.stylesheet
|
|
|
|
from sugar.graphics import style
|
2006-09-15 13:23:21 +02:00
|
|
|
from view.home.HomeWindow import HomeWindow
|
2006-09-15 15:28:18 +02:00
|
|
|
from sugar.presence import PresenceService
|
2006-09-15 13:23:21 +02:00
|
|
|
from view.ActivityHost import ActivityHost
|
2006-09-15 15:28:18 +02:00
|
|
|
from sugar.activity import ActivityFactory
|
|
|
|
from sugar.activity import Activity
|
2006-09-15 13:23:21 +02:00
|
|
|
from view.frame.Frame import Frame
|
2006-11-08 16:42:50 +01:00
|
|
|
from view.dconmanager import DCONManager
|
2006-10-12 12:47:17 +02:00
|
|
|
from _sugar import KeyGrabber
|
2007-01-05 02:19:52 +01:00
|
|
|
from _sugar import AudioManager
|
2006-08-30 11:15:21 +02:00
|
|
|
import sugar
|
2006-07-09 17:37:54 +02:00
|
|
|
|
2006-08-10 00:54:54 +02:00
|
|
|
class Shell(gobject.GObject):
|
2006-12-04 20:12:24 +01:00
|
|
|
def __init__(self, model):
|
|
|
|
gobject.GObject.__init__(self)
|
|
|
|
|
|
|
|
self._model = model
|
|
|
|
self._hosts = {}
|
|
|
|
self._screen = wnck.screen_get_default()
|
|
|
|
self._current_host = None
|
|
|
|
|
|
|
|
style.load_stylesheet(view.stylesheet)
|
|
|
|
|
|
|
|
self._dcon_manager = DCONManager()
|
2007-01-05 02:19:52 +01:00
|
|
|
self._audio_manager = AudioManager()
|
2006-12-04 20:12:24 +01:00
|
|
|
|
|
|
|
self._key_grabber = KeyGrabber()
|
|
|
|
self._key_grabber.connect('key-pressed',
|
2006-12-24 02:51:37 +01:00
|
|
|
self._key_pressed_cb)
|
2006-12-04 20:12:24 +01:00
|
|
|
self._key_grabber.connect('key-released',
|
2006-12-24 02:51:37 +01:00
|
|
|
self._key_released_cb)
|
|
|
|
self._grab_keys()
|
2006-12-04 20:12:24 +01:00
|
|
|
|
|
|
|
self._home_window = HomeWindow(self)
|
|
|
|
self._home_window.show()
|
|
|
|
self.set_zoom_level(sugar.ZOOM_HOME)
|
|
|
|
|
2006-12-24 14:35:02 +01:00
|
|
|
home_model = self._model.get_home()
|
|
|
|
home_model.connect('activity-added', self._activity_added_cb)
|
|
|
|
home_model.connect('activity-removed', self._activity_removed_cb)
|
2006-12-24 15:39:00 +01:00
|
|
|
home_model.connect('active-activity-changed',
|
|
|
|
self._active_activity_changed_cb)
|
2006-12-04 20:12:24 +01:00
|
|
|
|
|
|
|
self._frame = Frame(self)
|
|
|
|
self._frame.show_and_hide(3)
|
|
|
|
|
2007-01-06 22:29:13 +01:00
|
|
|
self._pservice = PresenceService.get_instance()
|
|
|
|
|
2007-01-05 02:19:52 +01:00
|
|
|
#self.start_activity('org.laptop.JournalActivity')
|
2006-12-24 02:59:20 +01:00
|
|
|
|
2006-12-21 12:37:02 +01:00
|
|
|
def _handle_camera_key(self):
|
|
|
|
if self._current_host:
|
|
|
|
if self._current_host.execute('camera', []):
|
|
|
|
return
|
|
|
|
|
|
|
|
self.start_activity('org.laptop.CameraActivity')
|
2006-12-04 20:12:24 +01:00
|
|
|
|
2006-12-24 02:51:37 +01:00
|
|
|
def _grab_keys(self):
|
|
|
|
self._key_grabber.grab('F1')
|
|
|
|
self._key_grabber.grab('F2')
|
|
|
|
self._key_grabber.grab('F3')
|
|
|
|
self._key_grabber.grab('F4')
|
|
|
|
self._key_grabber.grab('F5')
|
|
|
|
self._key_grabber.grab('F6')
|
|
|
|
self._key_grabber.grab('F7')
|
|
|
|
self._key_grabber.grab('F8')
|
2007-01-05 03:21:55 +01:00
|
|
|
self._key_grabber.grab('F9')
|
|
|
|
self._key_grabber.grab('F10')
|
|
|
|
self._key_grabber.grab('F11')
|
|
|
|
self._key_grabber.grab('F12')
|
2007-01-05 20:32:00 +01:00
|
|
|
self._key_grabber.grab('<alt>F5')
|
|
|
|
self._key_grabber.grab('<alt>F8')
|
2007-01-05 12:57:42 +01:00
|
|
|
|
2006-12-24 02:51:37 +01:00
|
|
|
self._key_grabber.grab('0xDC') # Camera key
|
|
|
|
self._key_grabber.grab('0xE0') # Overlay key
|
|
|
|
self._key_grabber.grab('0x93') # Frame key
|
2007-01-05 12:20:55 +01:00
|
|
|
self._key_grabber.grab('0x7C') # Power key
|
2006-12-24 12:19:24 +01:00
|
|
|
self._key_grabber.grab('<alt>Tab')
|
2006-12-24 02:51:37 +01:00
|
|
|
|
|
|
|
# For non-OLPC machines
|
|
|
|
self._key_grabber.grab('<shft><alt>F9')
|
|
|
|
self._key_grabber.grab('<shft><alt>F10')
|
|
|
|
|
|
|
|
def _key_pressed_cb(self, grabber, key):
|
2006-12-04 20:12:24 +01:00
|
|
|
if key == 'F1':
|
|
|
|
self.set_zoom_level(sugar.ZOOM_MESH)
|
|
|
|
elif key == 'F2':
|
|
|
|
self.set_zoom_level(sugar.ZOOM_FRIENDS)
|
|
|
|
elif key == 'F3':
|
|
|
|
self.set_zoom_level(sugar.ZOOM_HOME)
|
|
|
|
elif key == 'F4':
|
|
|
|
self.set_zoom_level(sugar.ZOOM_ACTIVITY)
|
|
|
|
elif key == 'F5':
|
2007-01-04 20:02:45 +01:00
|
|
|
self._dcon_manager.set_brightness(0)
|
2006-12-04 20:12:24 +01:00
|
|
|
elif key == 'F6':
|
2007-01-04 20:02:45 +01:00
|
|
|
self._dcon_manager.set_brightness(5)
|
2006-12-04 20:12:24 +01:00
|
|
|
elif key == 'F7':
|
2007-01-04 20:02:45 +01:00
|
|
|
self._dcon_manager.set_brightness(9)
|
2006-12-04 20:12:24 +01:00
|
|
|
elif key == 'F8':
|
2007-01-04 20:02:45 +01:00
|
|
|
self._dcon_manager.set_brightness(15)
|
2007-01-05 02:19:52 +01:00
|
|
|
elif key == 'F9':
|
|
|
|
self._audio_manager.set_volume(0)
|
|
|
|
elif key == 'F10':
|
|
|
|
self._audio_manager.set_volume(32)
|
|
|
|
elif key == 'F11':
|
|
|
|
self._audio_manager.set_volume(64)
|
|
|
|
elif key == 'F12':
|
|
|
|
self._audio_manager.set_volume(100)
|
2007-01-04 20:02:45 +01:00
|
|
|
elif key == '<alt>F5':
|
|
|
|
self._dcon_manager.set_mode(DCONManager.COLOR_MODE)
|
|
|
|
elif key == '<alt>F8':
|
2006-12-04 20:12:24 +01:00
|
|
|
self._dcon_manager.set_mode(DCONManager.BLACK_AND_WHITE_MODE)
|
|
|
|
elif key == '<shft><alt>F9':
|
|
|
|
self._frame.notify_key_press()
|
|
|
|
elif key == '<shft><alt>F10':
|
|
|
|
self.toggle_chat_visibility()
|
|
|
|
elif key == '0xDC': # Camera key
|
2006-12-21 12:37:02 +01:00
|
|
|
self._handle_camera_key()
|
2006-12-04 20:12:24 +01:00
|
|
|
elif key == '0xE0': # Overlay key
|
|
|
|
self.toggle_chat_visibility()
|
|
|
|
elif key == '0x93': # Frame key
|
|
|
|
self._frame.notify_key_press()
|
2007-01-05 12:20:55 +01:00
|
|
|
elif key == '0x7C': # Power key
|
2007-01-05 12:23:28 +01:00
|
|
|
self._shutdown()
|
2006-12-24 12:19:24 +01:00
|
|
|
elif key == '<alt>Tab':
|
|
|
|
self.set_zoom_level(sugar.ZOOM_HOME)
|
|
|
|
box = self._home_window.get_home_box()
|
|
|
|
box.grab_and_rotate()
|
2006-12-04 20:12:24 +01:00
|
|
|
|
2007-01-05 12:23:28 +01:00
|
|
|
def _shutdown(self):
|
|
|
|
bus = dbus.SystemBus()
|
|
|
|
proxy = bus.get_object('org.freedesktop.Hal',
|
|
|
|
'/org/freedesktop/Hal/devices/computer')
|
|
|
|
mgr = dbus.Interface(proxy, 'org.freedesktop.Hal.Device.SystemPowerManagement')
|
|
|
|
mgr.Shutdown()
|
|
|
|
|
2006-12-24 02:51:37 +01:00
|
|
|
def _key_released_cb(self, grabber, key):
|
2006-12-04 20:12:24 +01:00
|
|
|
if key == '<shft><alt>F9':
|
|
|
|
self._frame.notify_key_release()
|
|
|
|
elif key == '0x93':
|
|
|
|
self._frame.notify_key_release()
|
|
|
|
|
2006-12-24 14:35:02 +01:00
|
|
|
def _activity_added_cb(self, home_model, home_activity):
|
|
|
|
activity_host = ActivityHost(home_activity)
|
|
|
|
self._hosts[activity_host.get_xid()] = activity_host
|
2006-12-04 20:12:24 +01:00
|
|
|
|
2006-12-24 14:35:02 +01:00
|
|
|
def _activity_removed_cb(self, home_model, home_activity):
|
2006-12-24 15:39:00 +01:00
|
|
|
xid = home_activity.get_xid()
|
|
|
|
if self._hosts.has_key(xid):
|
|
|
|
self._hosts[xid].destroy()
|
|
|
|
del self._hosts[xid]
|
2006-12-04 20:12:24 +01:00
|
|
|
|
2006-12-24 15:39:00 +01:00
|
|
|
def _active_activity_changed_cb(self, home_model, home_activity):
|
2006-12-24 15:58:53 +01:00
|
|
|
if home_activity:
|
|
|
|
host = self._hosts[home_activity.get_xid()]
|
|
|
|
else:
|
|
|
|
host = None
|
2006-12-04 20:12:24 +01:00
|
|
|
|
|
|
|
if self._current_host:
|
|
|
|
self._current_host.set_active(False)
|
|
|
|
|
|
|
|
self._current_host = host
|
|
|
|
|
|
|
|
if self._current_host:
|
|
|
|
self._current_host.set_active(True)
|
|
|
|
|
|
|
|
def get_model(self):
|
|
|
|
return self._model
|
|
|
|
|
2007-01-07 06:04:30 +01:00
|
|
|
def _join_success_cb(self, handler, activity, activity_ps, activity_id, activity_type):
|
|
|
|
logging.debug("Joining activity %s (%s)" % (activity_id, activity_type))
|
|
|
|
activity.join(activity_ps.object_path())
|
|
|
|
|
|
|
|
def _join_error_cb(self, handler, err, home_model, activity_id, activity_type):
|
|
|
|
logging.error("Couldn't launch activity %s (%s):\n%s" % (activity_id, activity_type, err))
|
|
|
|
home_mode.notify_activity_launch_failed(activity_id)
|
|
|
|
|
2006-12-04 20:12:24 +01:00
|
|
|
def join_activity(self, bundle_id, activity_id):
|
2006-12-24 12:19:24 +01:00
|
|
|
activity = self.get_activity(activity_id)
|
2006-12-04 20:12:24 +01:00
|
|
|
if activity:
|
|
|
|
activity.present()
|
2007-01-07 06:04:30 +01:00
|
|
|
return
|
|
|
|
|
|
|
|
activity_ps = self._pservice.get_activity(activity_id)
|
|
|
|
if not activity_ps:
|
|
|
|
logging.error("Couldn't find shared activity for %s" % activity_id)
|
|
|
|
return
|
|
|
|
|
|
|
|
# Get the service name for this activity, if
|
|
|
|
# we have a bundle on the system capable of handling
|
|
|
|
# this activity type
|
|
|
|
breg = self._model.get_bundle_registry()
|
|
|
|
bundle = breg.find_by_default_type(bundle_id)
|
|
|
|
if not bundle:
|
|
|
|
logging.error("Couldn't find activity for type %s" % bundle_id)
|
|
|
|
return
|
|
|
|
|
|
|
|
act_type = bundle.get_service_name()
|
|
|
|
home_model = self._model.get_home()
|
|
|
|
home_model.notify_activity_launch(activity_id, act_type)
|
|
|
|
|
|
|
|
handler = ActivityFactory.create(act_type)
|
|
|
|
handler.connect('success', self._join_success_cb, activity_ps, activity_id, act_type)
|
|
|
|
handler.connect('error', self._join_error_cb, home_model, activity_id, act_type)
|
2006-12-04 20:12:24 +01:00
|
|
|
|
2007-01-06 22:29:13 +01:00
|
|
|
def _find_unique_activity_id(self):
|
|
|
|
# create a new unique activity ID
|
|
|
|
i = 0
|
|
|
|
act_id = None
|
|
|
|
while i < 10:
|
|
|
|
act_id = sugar.util.unique_id()
|
|
|
|
i += 1
|
|
|
|
|
|
|
|
# check through existing activities
|
|
|
|
found = False
|
|
|
|
for xid, act_host in self._hosts.items():
|
|
|
|
if act_host.get_id() == act_id:
|
|
|
|
found = True
|
|
|
|
break
|
|
|
|
if found:
|
|
|
|
act_id = None
|
|
|
|
continue
|
|
|
|
|
|
|
|
# check through network activities
|
|
|
|
activities = self._pservice.get_activities()
|
|
|
|
for act in activities:
|
|
|
|
if act_id == act.get_id():
|
|
|
|
found = True
|
|
|
|
break
|
|
|
|
if found:
|
|
|
|
act_id = None
|
|
|
|
continue
|
|
|
|
|
|
|
|
return act_id
|
|
|
|
|
2007-01-07 06:04:30 +01:00
|
|
|
def _start_success_cb(self, handler, activity, activity_id, activity_type):
|
|
|
|
logging.debug("Started activity %s (%s)" % (activity_id, activity_type))
|
|
|
|
activity.start(activity_id)
|
|
|
|
|
|
|
|
def _start_error_cb(self, handler, err, home_model, activity_id, activity_type):
|
|
|
|
logging.error("Couldn't launch activity %s (%s):\n%s" % (activity_id, activity_type, err))
|
|
|
|
home_mode.notify_activity_launch_failed(activity_id)
|
|
|
|
|
2006-12-04 20:12:24 +01:00
|
|
|
def start_activity(self, activity_type):
|
2006-12-16 23:55:22 +01:00
|
|
|
logging.debug('Shell.start_activity')
|
2007-01-06 22:29:13 +01:00
|
|
|
act_id = self._find_unique_activity_id()
|
|
|
|
if not act_id:
|
|
|
|
logging.error("Couldn't find available activity ID.")
|
|
|
|
return None
|
|
|
|
|
2007-01-07 01:31:19 +01:00
|
|
|
home_model = self._model.get_home()
|
|
|
|
home_model.notify_activity_launch(act_id, activity_type)
|
|
|
|
|
2007-01-07 06:04:30 +01:00
|
|
|
logging.debug("Shell.start_activity will start %s (%s)" % (act_id, activity_type))
|
|
|
|
handler = ActivityFactory.create(activity_type)
|
|
|
|
handler.connect('success', self._start_success_cb, act_id, activity_type)
|
|
|
|
handler.connect('error', self._start_error_cb, home_model, act_id, activity_type)
|
2006-12-04 20:12:24 +01:00
|
|
|
|
|
|
|
def set_zoom_level(self, level):
|
|
|
|
if level == sugar.ZOOM_ACTIVITY:
|
|
|
|
self._screen.toggle_showing_desktop(False)
|
|
|
|
else:
|
|
|
|
self._screen.toggle_showing_desktop(True)
|
|
|
|
self._home_window.set_zoom_level(level)
|
|
|
|
|
|
|
|
def get_current_activity(self):
|
2006-12-24 15:39:00 +01:00
|
|
|
return self._current_host
|
2006-12-04 20:12:24 +01:00
|
|
|
|
2006-12-24 12:19:24 +01:00
|
|
|
def get_activity(self, activity_id):
|
2006-12-04 20:12:24 +01:00
|
|
|
for host in self._hosts.values():
|
|
|
|
if host.get_id() == activity_id:
|
|
|
|
return host
|
|
|
|
return None
|
|
|
|
|
|
|
|
def toggle_chat_visibility(self):
|
|
|
|
act = self.get_current_activity()
|
|
|
|
if not act:
|
|
|
|
return
|
|
|
|
is_visible = self._frame.is_visible()
|
|
|
|
if act.is_chat_visible():
|
|
|
|
frame_was_visible = act.chat_hide()
|
|
|
|
if not frame_was_visible:
|
|
|
|
self._frame.do_slide_out()
|
|
|
|
else:
|
|
|
|
if not is_visible:
|
|
|
|
self._frame.do_slide_in()
|
|
|
|
act.chat_show(is_visible)
|