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
|
|
|
|
|
2007-06-04 19:35:05 +02:00
|
|
|
from gettext import gettext as _
|
2007-03-15 16:21:37 +01:00
|
|
|
from sets import Set
|
2006-12-14 16:45:52 +01:00
|
|
|
import logging
|
2007-06-04 19:35:05 +02:00
|
|
|
import tempfile
|
|
|
|
import os
|
|
|
|
import time
|
2006-12-14 16:45:52 +01:00
|
|
|
|
2006-07-12 22:17:57 +02:00
|
|
|
import gobject
|
2007-06-04 19:35:05 +02:00
|
|
|
import gtk
|
2006-07-27 10:35:59 +02:00
|
|
|
import wnck
|
2006-07-06 23:34:23 +02:00
|
|
|
|
2007-02-22 00:07:08 +01:00
|
|
|
from sugar.activity.activityhandle import ActivityHandle
|
2007-02-21 10:16:03 +01:00
|
|
|
from sugar.graphics.popupcontext import PopupContext
|
2007-05-27 20:24:10 +02:00
|
|
|
from sugar.activity import activityfactory
|
2007-06-04 19:35:05 +02:00
|
|
|
from sugar.datastore import datastore
|
|
|
|
from sugar import profile
|
2007-05-27 20:24:10 +02:00
|
|
|
import sugar
|
|
|
|
|
2006-09-15 13:23:21 +02:00
|
|
|
from view.ActivityHost import ActivityHost
|
2007-01-25 12:22:37 +01:00
|
|
|
from view.frame.frame import Frame
|
2007-01-18 15:04:56 +01:00
|
|
|
from view.keyhandler import KeyHandler
|
2007-05-27 20:24:10 +02:00
|
|
|
from view.home.HomeWindow import HomeWindow
|
|
|
|
from model import bundleregistry
|
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)
|
|
|
|
|
2007-03-15 16:21:37 +01:00
|
|
|
self._activities_starting = Set()
|
2006-12-04 20:12:24 +01:00
|
|
|
self._model = model
|
|
|
|
self._hosts = {}
|
|
|
|
self._screen = wnck.screen_get_default()
|
|
|
|
self._current_host = None
|
2007-01-13 23:08:21 +01:00
|
|
|
self._screen_rotation = 0
|
2006-12-04 20:12:24 +01:00
|
|
|
|
2007-03-12 20:18:55 +01:00
|
|
|
self._key_handler = KeyHandler(self)
|
|
|
|
self._popup_context = PopupContext()
|
2007-03-12 22:29:11 +01:00
|
|
|
|
2007-03-12 20:18:55 +01:00
|
|
|
self._frame = Frame(self)
|
2007-03-12 22:29:11 +01:00
|
|
|
self._frame.show()
|
2007-03-12 20:18:55 +01:00
|
|
|
|
2006-12-04 20:12:24 +01:00
|
|
|
self._home_window = HomeWindow(self)
|
|
|
|
self._home_window.show()
|
2007-03-12 22:29:11 +01:00
|
|
|
|
2007-03-12 20:18:55 +01:00
|
|
|
self._zoom_level = sugar.ZOOM_HOME
|
2007-01-18 15:04:56 +01:00
|
|
|
|
2006-12-24 14:35:02 +01:00
|
|
|
home_model = self._model.get_home()
|
2007-06-01 11:03:18 +02:00
|
|
|
home_model.connect('activity-started', self._activity_started_cb)
|
2006-12-24 14:35:02 +01:00
|
|
|
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
|
|
|
|
2007-04-29 21:12:35 +02:00
|
|
|
self.start_activity('org.laptop.JournalActivity')
|
2006-12-24 02:59:20 +01:00
|
|
|
|
2007-06-01 11:03:18 +02:00
|
|
|
def _activity_started_cb(self, home_model, home_activity):
|
2006-12-24 14:35:02 +01:00
|
|
|
activity_host = ActivityHost(home_activity)
|
|
|
|
self._hosts[activity_host.get_xid()] = activity_host
|
2007-03-15 16:21:37 +01:00
|
|
|
if home_activity.get_type() in self._activities_starting:
|
|
|
|
self._activities_starting.remove(home_activity.get_type())
|
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):
|
2007-03-15 16:21:37 +01:00
|
|
|
if home_activity.get_type() in self._activities_starting:
|
|
|
|
self._activities_starting.remove(home_activity.get_type())
|
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)
|
2007-03-13 00:19:21 +01:00
|
|
|
self.set_zoom_level(sugar.ZOOM_ACTIVITY)
|
2007-03-12 22:47:17 +01:00
|
|
|
else:
|
|
|
|
self.set_zoom_level(sugar.ZOOM_HOME)
|
2007-03-12 22:29:11 +01:00
|
|
|
|
2006-12-04 20:12:24 +01:00
|
|
|
def get_model(self):
|
|
|
|
return self._model
|
|
|
|
|
2007-01-18 15:04:56 +01:00
|
|
|
def get_frame(self):
|
|
|
|
return self._frame
|
|
|
|
|
2007-02-21 10:16:03 +01:00
|
|
|
def get_popup_context(self):
|
|
|
|
return self._popup_context
|
2007-02-20 16:38:25 +01:00
|
|
|
|
2007-02-21 20:56:14 +01:00
|
|
|
def _join_error_cb(self, handler, err, home_model):
|
2007-05-03 04:31:26 +02:00
|
|
|
logging.debug("Failed to join activity %s: %s" % (handler.get_activity_id(), err))
|
|
|
|
home_model.notify_activity_launch_failed(handler.get_activity_id())
|
2007-01-07 06:04:30 +01:00
|
|
|
|
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
|
|
|
|
|
|
|
|
# Get the service name for this activity, if
|
|
|
|
# we have a bundle on the system capable of handling
|
|
|
|
# this activity type
|
2007-05-03 04:21:15 +02:00
|
|
|
breg = bundleregistry.get_registry()
|
2007-04-11 21:08:40 +02:00
|
|
|
bundle = breg.get_bundle(bundle_id)
|
2007-01-07 06:04:30 +01:00
|
|
|
if not bundle:
|
|
|
|
logging.error("Couldn't find activity for type %s" % bundle_id)
|
|
|
|
return
|
|
|
|
|
|
|
|
home_model = self._model.get_home()
|
2007-04-11 21:08:40 +02:00
|
|
|
home_model.notify_activity_launch(activity_id, bundle_id)
|
2007-01-07 06:04:30 +01:00
|
|
|
|
2007-02-22 00:07:08 +01:00
|
|
|
handle = ActivityHandle(activity_id)
|
|
|
|
handle.pservice_id = activity_id
|
|
|
|
|
2007-04-11 21:08:40 +02:00
|
|
|
handler = activityfactory.create(bundle_id, handle)
|
2007-02-21 20:56:14 +01:00
|
|
|
handler.connect('error', self._join_error_cb, home_model)
|
|
|
|
|
2007-02-22 00:57:49 +01:00
|
|
|
def _start_error_cb(self, handler, err, home_model):
|
2007-02-21 20:56:14 +01:00
|
|
|
home_model.notify_activity_launch_failed(handler.get_activity_id())
|
2007-01-07 06:04:30 +01:00
|
|
|
|
2006-12-04 20:12:24 +01:00
|
|
|
def start_activity(self, activity_type):
|
2007-03-15 16:21:37 +01:00
|
|
|
if activity_type in self._activities_starting:
|
|
|
|
logging.debug("This activity is still launching.")
|
|
|
|
return
|
|
|
|
|
2007-05-03 04:22:02 +02:00
|
|
|
logging.debug('Trying to start activity of type %s' % activity_type)
|
2007-02-21 20:56:14 +01:00
|
|
|
|
2007-03-15 16:21:37 +01:00
|
|
|
self._activities_starting.add(activity_type)
|
2007-03-16 14:04:49 +01:00
|
|
|
try:
|
|
|
|
handler = activityfactory.create(activity_type)
|
2007-03-15 16:21:37 +01:00
|
|
|
|
2007-03-16 14:04:49 +01:00
|
|
|
home_model = self._model.get_home()
|
|
|
|
home_model.notify_activity_launch(handler.get_activity_id(),
|
|
|
|
activity_type)
|
2007-01-06 22:29:13 +01:00
|
|
|
|
2007-03-16 14:04:49 +01:00
|
|
|
handler.connect('error', self._start_error_cb, home_model)
|
2007-05-03 04:22:02 +02:00
|
|
|
except Exception, err:
|
|
|
|
logging.debug("Couldn't start activity of type %s: %s" % (activity_type, err))
|
2007-03-16 14:04:49 +01:00
|
|
|
self._activities_starting.remove(activity_type)
|
2006-12-04 20:12:24 +01:00
|
|
|
|
2007-01-08 23:05:55 +01:00
|
|
|
# Zoom to Home for launch feedback
|
|
|
|
self.set_zoom_level(sugar.ZOOM_HOME)
|
|
|
|
|
2006-12-04 20:12:24 +01:00
|
|
|
def set_zoom_level(self, level):
|
2007-03-12 22:29:11 +01:00
|
|
|
if self._zoom_level == level:
|
|
|
|
return
|
|
|
|
if len(self._hosts) == 0 and level == sugar.ZOOM_ACTIVITY:
|
|
|
|
return
|
|
|
|
|
2007-03-12 20:18:55 +01:00
|
|
|
self._zoom_level = level
|
|
|
|
|
2007-03-12 22:29:11 +01:00
|
|
|
if self._zoom_level == sugar.ZOOM_ACTIVITY:
|
2006-12-04 20:12:24 +01:00
|
|
|
self._screen.toggle_showing_desktop(False)
|
|
|
|
else:
|
|
|
|
self._screen.toggle_showing_desktop(True)
|
2007-03-12 22:29:11 +01:00
|
|
|
self._home_window.set_zoom_level(self._zoom_level)
|
|
|
|
|
|
|
|
if self._zoom_level == sugar.ZOOM_HOME:
|
|
|
|
self._frame.show()
|
2007-03-13 00:19:21 +01:00
|
|
|
else:
|
|
|
|
self._frame.hide()
|
2006-12-04 20:12:24 +01:00
|
|
|
|
|
|
|
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)
|
2007-06-04 19:35:05 +02:00
|
|
|
|
|
|
|
def take_screenshot(self):
|
|
|
|
file_path = os.path.join(tempfile.gettempdir(), '%i' % time.time())
|
|
|
|
|
|
|
|
window = gtk.gdk.get_default_root_window()
|
|
|
|
width, height = window.get_size();
|
|
|
|
x_orig, y_orig = window.get_origin();
|
|
|
|
|
|
|
|
screenshot = gtk.gdk.Pixbuf(gtk.gdk.COLORSPACE_RGB, has_alpha=False,
|
|
|
|
bits_per_sample=8, width=width, height=height)
|
|
|
|
screenshot.get_from_drawable(window, window.get_colormap(), x_orig, y_orig, 0, 0,
|
|
|
|
width, height);
|
|
|
|
screenshot.save(file_path, "png")
|
|
|
|
|
|
|
|
jobject = datastore.create()
|
|
|
|
jobject.metadata['title'] = _('Screenshot')
|
|
|
|
jobject.metadata['keep'] = '0'
|
|
|
|
jobject.metadata['buddies'] = ''
|
|
|
|
jobject.metadata['preview'] = ''
|
|
|
|
jobject.metadata['icon-color'] = profile.get_color().to_string()
|
2007-06-12 12:40:54 +02:00
|
|
|
jobject.metadata['mime_type'] = 'image/png'
|
2007-06-04 19:35:05 +02:00
|
|
|
jobject.file_path = file_path
|
|
|
|
datastore.write(jobject)
|
|
|
|
|