Split model/view in the activities donut.
Provide some infrastructure for Alt+Tab implementation.
This commit is contained in:
parent
db6f615198
commit
2db2ae5312
@ -6,4 +6,6 @@ sugar_PYTHON = \
|
|||||||
Invites.py \
|
Invites.py \
|
||||||
Owner.py \
|
Owner.py \
|
||||||
MeshModel.py \
|
MeshModel.py \
|
||||||
ShellModel.py
|
ShellModel.py \
|
||||||
|
homeactivity.py \
|
||||||
|
homemodel.py
|
||||||
|
32
shell/model/homeactivity.py
Normal file
32
shell/model/homeactivity.py
Normal file
@ -0,0 +1,32 @@
|
|||||||
|
# Copyright (C) 2006, Owen Williams.
|
||||||
|
#
|
||||||
|
# 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
|
||||||
|
|
||||||
|
from sugar.graphics.canvasicon import CanvasIcon
|
||||||
|
|
||||||
|
class HomeActivity:
|
||||||
|
def __init__(self, activity):
|
||||||
|
self._icon_name = activity.get_icon_name()
|
||||||
|
self._icon_color = activity.get_icon_color()
|
||||||
|
self._id = activity.get_id()
|
||||||
|
|
||||||
|
def get_icon_name(self):
|
||||||
|
return self._icon_name
|
||||||
|
|
||||||
|
def get_icon_color(self):
|
||||||
|
return self._icon_color
|
||||||
|
|
||||||
|
def get_id(self):
|
||||||
|
return self._id
|
69
shell/model/homemodel.py
Normal file
69
shell/model/homemodel.py
Normal file
@ -0,0 +1,69 @@
|
|||||||
|
# Copyright (C) 2006, Owen Williams.
|
||||||
|
#
|
||||||
|
# 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
|
||||||
|
|
||||||
|
import gobject
|
||||||
|
|
||||||
|
from sugar.graphics.canvasicon import CanvasIcon
|
||||||
|
from sugar.graphics import style
|
||||||
|
from model.homeactivity import HomeActivity
|
||||||
|
|
||||||
|
class HomeModel(gobject.GObject):
|
||||||
|
|
||||||
|
__gsignals__ = {
|
||||||
|
'activity-added': (gobject.SIGNAL_RUN_FIRST,
|
||||||
|
gobject.TYPE_NONE,
|
||||||
|
([gobject.TYPE_PYOBJECT])),
|
||||||
|
'activity-removed': (gobject.SIGNAL_RUN_FIRST,
|
||||||
|
gobject.TYPE_NONE,
|
||||||
|
([gobject.TYPE_PYOBJECT]))
|
||||||
|
}
|
||||||
|
|
||||||
|
def __init__(self, shell):
|
||||||
|
gobject.GObject.__init__(self)
|
||||||
|
self._activities = []
|
||||||
|
self._shell = shell
|
||||||
|
|
||||||
|
shell.connect('activity-opened', self.__activity_opened_cb)
|
||||||
|
shell.connect('activity-closed', self.__activity_closed_cb)
|
||||||
|
|
||||||
|
def __iter__(self):
|
||||||
|
return iter(self._activities)
|
||||||
|
|
||||||
|
def __len__(self):
|
||||||
|
return len(self._activities)
|
||||||
|
|
||||||
|
def __getitem__(self, i):
|
||||||
|
return self._activities[i]
|
||||||
|
|
||||||
|
def __activity_opened_cb(self, model, activity):
|
||||||
|
self._add_activity(activity)
|
||||||
|
|
||||||
|
def __activity_closed_cb(self, model, activity):
|
||||||
|
self._remove_activity(activity)
|
||||||
|
|
||||||
|
def _add_activity(self, activity):
|
||||||
|
h_activity = HomeActivity(activity)
|
||||||
|
self._activities.append(h_activity)
|
||||||
|
self.emit('activity-added', h_activity)
|
||||||
|
|
||||||
|
def _remove_activity(self, activity):
|
||||||
|
i = 0
|
||||||
|
for h_activity in self._activities:
|
||||||
|
if h_activity.get_id() == activity.get_id():
|
||||||
|
self.emit('activity-removed', self._activities[i])
|
||||||
|
del self._activities[i]
|
||||||
|
return
|
||||||
|
i += 1
|
@ -95,6 +95,7 @@ class Shell(gobject.GObject):
|
|||||||
self._key_grabber.grab('0xDC') # Camera key
|
self._key_grabber.grab('0xDC') # Camera key
|
||||||
self._key_grabber.grab('0xE0') # Overlay key
|
self._key_grabber.grab('0xE0') # Overlay key
|
||||||
self._key_grabber.grab('0x93') # Frame key
|
self._key_grabber.grab('0x93') # Frame key
|
||||||
|
self._key_grabber.grab('<alt>Tab')
|
||||||
|
|
||||||
# For non-OLPC machines
|
# For non-OLPC machines
|
||||||
self._key_grabber.grab('<shft><alt>F9')
|
self._key_grabber.grab('<shft><alt>F9')
|
||||||
@ -127,6 +128,10 @@ class Shell(gobject.GObject):
|
|||||||
self.toggle_chat_visibility()
|
self.toggle_chat_visibility()
|
||||||
elif key == '0x93': # Frame key
|
elif key == '0x93': # Frame key
|
||||||
self._frame.notify_key_press()
|
self._frame.notify_key_press()
|
||||||
|
elif key == '<alt>Tab':
|
||||||
|
self.set_zoom_level(sugar.ZOOM_HOME)
|
||||||
|
box = self._home_window.get_home_box()
|
||||||
|
box.grab_and_rotate()
|
||||||
|
|
||||||
def _key_released_cb(self, grabber, key):
|
def _key_released_cb(self, grabber, key):
|
||||||
if key == '<shft><alt>F9':
|
if key == '<shft><alt>F9':
|
||||||
@ -202,7 +207,7 @@ class Shell(gobject.GObject):
|
|||||||
def join_activity(self, bundle_id, activity_id):
|
def join_activity(self, bundle_id, activity_id):
|
||||||
pservice = PresenceService.get_instance()
|
pservice = PresenceService.get_instance()
|
||||||
|
|
||||||
activity = self._get_activity(activity_id)
|
activity = self.get_activity(activity_id)
|
||||||
if activity:
|
if activity:
|
||||||
activity.present()
|
activity.present()
|
||||||
else:
|
else:
|
||||||
@ -244,11 +249,11 @@ class Shell(gobject.GObject):
|
|||||||
def get_current_activity(self):
|
def get_current_activity(self):
|
||||||
activity_id = self._model.get_current_activity()
|
activity_id = self._model.get_current_activity()
|
||||||
if activity_id:
|
if activity_id:
|
||||||
return self._get_activity(activity_id)
|
return self.get_activity(activity_id)
|
||||||
else:
|
else:
|
||||||
return None
|
return None
|
||||||
|
|
||||||
def _get_activity(self, activity_id):
|
def get_activity(self, activity_id):
|
||||||
for host in self._hosts.values():
|
for host in self._hosts.values():
|
||||||
if host.get_id() == activity_id:
|
if host.get_id() == activity_id:
|
||||||
return host
|
return host
|
||||||
|
@ -25,13 +25,12 @@ class HomeBox(hippo.CanvasBox, hippo.CanvasItem):
|
|||||||
__gtype_name__ = 'SugarHomeBox'
|
__gtype_name__ = 'SugarHomeBox'
|
||||||
|
|
||||||
def __init__(self, shell):
|
def __init__(self, shell):
|
||||||
hippo.CanvasBox.__init__(self, background_color=0xe2e2e2ff,
|
hippo.CanvasBox.__init__(self, background_color=0xe2e2e2ff, yalign=2)
|
||||||
yalign=2)
|
|
||||||
|
|
||||||
grid = Grid()
|
grid = Grid()
|
||||||
donut = ActivitiesDonut(shell, box_width=grid.dimension(7),
|
self._donut = ActivitiesDonut(shell, box_width=grid.dimension(7),
|
||||||
box_height=grid.dimension(7))
|
box_height=grid.dimension(7))
|
||||||
self.append(donut)
|
self.append(self._donut)
|
||||||
|
|
||||||
self._my_icon = MyIcon()
|
self._my_icon = MyIcon()
|
||||||
style.apply_stylesheet(self._my_icon, 'home.MyIcon')
|
style.apply_stylesheet(self._my_icon, 'home.MyIcon')
|
||||||
@ -43,3 +42,15 @@ class HomeBox(hippo.CanvasBox, hippo.CanvasItem):
|
|||||||
[icon_width, icon_height] = self._my_icon.get_allocation()
|
[icon_width, icon_height] = self._my_icon.get_allocation()
|
||||||
self.move(self._my_icon, (width - icon_width) / 2,
|
self.move(self._my_icon, (width - icon_width) / 2,
|
||||||
(height - icon_height) / 2)
|
(height - icon_height) / 2)
|
||||||
|
|
||||||
|
def has_activities(self):
|
||||||
|
return self._donut.has_activities()
|
||||||
|
|
||||||
|
def grab_and_rotate(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def rotate(self):
|
||||||
|
pass
|
||||||
|
|
||||||
|
def release(self):
|
||||||
|
pass
|
||||||
|
@ -34,6 +34,7 @@ class HomeWindow(gtk.Window):
|
|||||||
|
|
||||||
self.realize()
|
self.realize()
|
||||||
self.window.set_type_hint(gtk.gdk.WINDOW_TYPE_HINT_DESKTOP)
|
self.window.set_type_hint(gtk.gdk.WINDOW_TYPE_HINT_DESKTOP)
|
||||||
|
self.connect("key-release-event", self._key_release_cb)
|
||||||
|
|
||||||
self._nb = gtk.Notebook()
|
self._nb = gtk.Notebook()
|
||||||
self._nb.set_show_border(False)
|
self._nb.set_show_border(False)
|
||||||
@ -43,8 +44,8 @@ class HomeWindow(gtk.Window):
|
|||||||
self._nb.show()
|
self._nb.show()
|
||||||
|
|
||||||
canvas = hippo.Canvas()
|
canvas = hippo.Canvas()
|
||||||
box = HomeBox(shell)
|
self._home_box = HomeBox(shell)
|
||||||
canvas.set_root(box)
|
canvas.set_root(self._home_box)
|
||||||
self._nb.append_page(canvas)
|
self._nb.append_page(canvas)
|
||||||
canvas.show()
|
canvas.show()
|
||||||
|
|
||||||
@ -60,6 +61,11 @@ class HomeWindow(gtk.Window):
|
|||||||
self._nb.append_page(canvas)
|
self._nb.append_page(canvas)
|
||||||
canvas.show()
|
canvas.show()
|
||||||
|
|
||||||
|
def _key_release_cb(self, widget, event):
|
||||||
|
keyname = gtk.gdk.keyval_name(event.keyval)
|
||||||
|
if keyname == "Alt_L":
|
||||||
|
self._home_box.release()
|
||||||
|
|
||||||
def set_zoom_level(self, level):
|
def set_zoom_level(self, level):
|
||||||
if level == sugar.ZOOM_HOME:
|
if level == sugar.ZOOM_HOME:
|
||||||
self._nb.set_current_page(0)
|
self._nb.set_current_page(0)
|
||||||
@ -67,3 +73,6 @@ class HomeWindow(gtk.Window):
|
|||||||
self._nb.set_current_page(1)
|
self._nb.set_current_page(1)
|
||||||
elif level == sugar.ZOOM_MESH:
|
elif level == sugar.ZOOM_MESH:
|
||||||
self._nb.set_current_page(2)
|
self._nb.set_current_page(2)
|
||||||
|
|
||||||
|
def get_home_box(self):
|
||||||
|
return self._home_box
|
||||||
|
@ -19,6 +19,7 @@ import math
|
|||||||
|
|
||||||
from sugar.graphics.canvasicon import CanvasIcon
|
from sugar.graphics.canvasicon import CanvasIcon
|
||||||
from sugar.graphics import style
|
from sugar.graphics import style
|
||||||
|
from model.homemodel import HomeModel
|
||||||
|
|
||||||
class ActivitiesDonut(hippo.CanvasBox, hippo.CanvasItem):
|
class ActivitiesDonut(hippo.CanvasBox, hippo.CanvasItem):
|
||||||
__gtype_name__ = 'SugarActivitiesDonut'
|
__gtype_name__ = 'SugarActivitiesDonut'
|
||||||
@ -26,14 +27,16 @@ class ActivitiesDonut(hippo.CanvasBox, hippo.CanvasItem):
|
|||||||
hippo.CanvasBox.__init__(self, **kwargs)
|
hippo.CanvasBox.__init__(self, **kwargs)
|
||||||
|
|
||||||
self._activities = {}
|
self._activities = {}
|
||||||
|
self._shell = shell
|
||||||
|
|
||||||
shell.connect('activity_opened', self.__activity_opened_cb)
|
self._model = HomeModel(shell)
|
||||||
shell.connect('activity_closed', self.__activity_closed_cb)
|
self._model.connect('activity-added', self._activity_added_cb)
|
||||||
|
self._model.connect('activity-removed', self._activity_removed_cb)
|
||||||
|
|
||||||
def __activity_opened_cb(self, model, activity):
|
def _activity_added_cb(self, model, activity):
|
||||||
self._add_activity(activity)
|
self._add_activity(activity)
|
||||||
|
|
||||||
def __activity_closed_cb(self, model, activity):
|
def _activity_removed_cb(self, model, activity):
|
||||||
self._remove_activity(activity)
|
self._remove_activity(activity)
|
||||||
|
|
||||||
def _remove_activity(self, activity):
|
def _remove_activity(self, activity):
|
||||||
@ -47,15 +50,17 @@ class ActivitiesDonut(hippo.CanvasBox, hippo.CanvasItem):
|
|||||||
|
|
||||||
icon = CanvasIcon(icon_name=icon_name, color=icon_color)
|
icon = CanvasIcon(icon_name=icon_name, color=icon_color)
|
||||||
style.apply_stylesheet(icon, 'ring.ActivityIcon')
|
style.apply_stylesheet(icon, 'ring.ActivityIcon')
|
||||||
icon.connect('activated', self.__activity_icon_clicked_cb, activity)
|
icon.connect('activated', self._activity_icon_clicked_cb, activity)
|
||||||
self.append(icon, hippo.PACK_FIXED)
|
self.append(icon, hippo.PACK_FIXED)
|
||||||
|
|
||||||
self._activities[activity.get_id()] = icon
|
self._activities[activity.get_id()] = icon
|
||||||
|
|
||||||
self.emit_paint_needed(0, 0, -1, -1)
|
self.emit_paint_needed(0, 0, -1, -1)
|
||||||
|
|
||||||
def __activity_icon_clicked_cb(self, item, activity):
|
def _activity_icon_clicked_cb(self, item, activity):
|
||||||
activity.present()
|
activity_host = self._shell.get_activity(activity.get_id())
|
||||||
|
if activity_host:
|
||||||
|
activity_host.present()
|
||||||
|
|
||||||
def _get_angles(self, index):
|
def _get_angles(self, index):
|
||||||
angle = 2 * math.pi / 8
|
angle = 2 * math.pi / 8
|
||||||
|
@ -56,7 +56,7 @@ class Timeline:
|
|||||||
del self._tags[name]
|
del self._tags[name]
|
||||||
|
|
||||||
def _next_frame(self, tag, frame):
|
def _next_frame(self, tag, frame):
|
||||||
n_frames = tag.start_frame - tag.end_frame
|
n_frames = tag.end_frame - tag.start_frame + 1
|
||||||
self._observer.next_frame(tag.name, frame, n_frames)
|
self._observer.next_frame(tag.name, frame, n_frames)
|
||||||
|
|
||||||
def goto(self, tag_name, end_frame=False):
|
def goto(self, tag_name, end_frame=False):
|
||||||
|
Loading…
Reference in New Issue
Block a user