462 lines
15 KiB
Python
462 lines
15 KiB
Python
# Copyright (C) 2007, One Laptop Per Child
|
|
#
|
|
# This library is free software; you can redistribute it and/or
|
|
# modify it under the terms of the GNU Lesser General Public
|
|
# License as published by the Free Software Foundation; either
|
|
# version 2 of the License, or (at your option) any later version.
|
|
#
|
|
# This library 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
|
|
# Lesser General Public License for more details.
|
|
#
|
|
# You should have received a copy of the GNU Lesser General Public
|
|
# License along with this library; if not, write to the
|
|
# Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
# Boston, MA 02111-1307, USA.
|
|
|
|
"""
|
|
STABLE.
|
|
"""
|
|
|
|
import gobject
|
|
import gtk
|
|
|
|
from sugar.graphics import style
|
|
from sugar.graphics.palette import ToolInvoker
|
|
from sugar.graphics.toolbutton import ToolButton
|
|
from sugar.graphics.icon import Icon
|
|
|
|
_PREVIOUS_PAGE = 0
|
|
_NEXT_PAGE = 1
|
|
|
|
class _TrayViewport(gtk.Viewport):
|
|
__gproperties__ = {
|
|
'scrollable' : (bool, None, None, False,
|
|
gobject.PARAM_READABLE),
|
|
'can-scroll-prev' : (bool, None, None, False,
|
|
gobject.PARAM_READABLE),
|
|
'can-scroll-next' : (bool, None, None, False,
|
|
gobject.PARAM_READABLE),
|
|
}
|
|
|
|
def __init__(self, orientation):
|
|
self.orientation = orientation
|
|
self._scrollable = False
|
|
self._can_scroll_next = False
|
|
self._can_scroll_prev = False
|
|
|
|
gobject.GObject.__init__(self)
|
|
|
|
self.set_shadow_type(gtk.SHADOW_NONE)
|
|
|
|
self.traybar = gtk.Toolbar()
|
|
self.traybar.set_orientation(orientation)
|
|
self.traybar.set_show_arrow(False)
|
|
self.add(self.traybar)
|
|
self.traybar.show()
|
|
|
|
self.connect('size_allocate', self._size_allocate_cb)
|
|
|
|
if self.orientation == gtk.ORIENTATION_HORIZONTAL:
|
|
adj = self.get_hadjustment()
|
|
else:
|
|
adj = self.get_vadjustment()
|
|
adj.connect('changed', self._adjustment_changed_cb)
|
|
adj.connect('value-changed', self._adjustment_changed_cb)
|
|
|
|
def scroll(self, direction):
|
|
if direction == _PREVIOUS_PAGE:
|
|
self._scroll_previous()
|
|
elif direction == _NEXT_PAGE:
|
|
self._scroll_next()
|
|
|
|
def scroll_to_item(self, item):
|
|
"""This function scrolls the viewport so that item will be visible."""
|
|
assert item in self.traybar.get_children()
|
|
|
|
# Get the allocation, and make sure that it is visible
|
|
if self.orientation == gtk.ORIENTATION_HORIZONTAL:
|
|
adj = self.get_hadjustment()
|
|
start = item.allocation.x
|
|
stop = item.allocation.x + item.allocation.width
|
|
else:
|
|
adj = self.get_vadjustment()
|
|
start = item.allocation.y
|
|
stop = item.allocation.y + item.allocation.height
|
|
|
|
if start < adj.value:
|
|
adj.value = start
|
|
elif stop > adj.value + adj.page_size:
|
|
adj.value = stop - adj.page_size
|
|
|
|
def _scroll_next(self):
|
|
allocation = self.get_allocation()
|
|
if self.orientation == gtk.ORIENTATION_HORIZONTAL:
|
|
adj = self.get_hadjustment()
|
|
new_value = adj.value + allocation.width
|
|
adj.value = min(new_value, adj.upper - allocation.width)
|
|
else:
|
|
adj = self.get_vadjustment()
|
|
new_value = adj.value + allocation.height
|
|
adj.value = min(new_value, adj.upper - allocation.height)
|
|
|
|
def _scroll_previous(self):
|
|
allocation = self.get_allocation()
|
|
if self.orientation == gtk.ORIENTATION_HORIZONTAL:
|
|
adj = self.get_hadjustment()
|
|
new_value = adj.value - allocation.width
|
|
adj.value = max(adj.lower, new_value)
|
|
else:
|
|
adj = self.get_vadjustment()
|
|
new_value = adj.value - allocation.height
|
|
adj.value = max(adj.lower, new_value)
|
|
|
|
def do_size_request(self, requisition):
|
|
child_requisition = self.get_child().size_request()
|
|
if self.orientation == gtk.ORIENTATION_HORIZONTAL:
|
|
requisition[0] = 0
|
|
requisition[1] = child_requisition[1]
|
|
else:
|
|
requisition[0] = child_requisition[0]
|
|
requisition[1] = 0
|
|
|
|
def do_get_property(self, pspec):
|
|
if pspec.name == 'scrollable':
|
|
return self._scrollable
|
|
elif pspec.name == 'can-scroll-next':
|
|
return self._can_scroll_next
|
|
elif pspec.name == 'can-scroll-prev':
|
|
return self._can_scroll_prev
|
|
|
|
def _size_allocate_cb(self, viewport, allocation):
|
|
bar_requisition = self.traybar.get_child_requisition()
|
|
if self.orientation == gtk.ORIENTATION_HORIZONTAL:
|
|
scrollable = bar_requisition[0] > allocation.width
|
|
else:
|
|
scrollable = bar_requisition[1] > allocation.height
|
|
|
|
if scrollable != self._scrollable:
|
|
self._scrollable = scrollable
|
|
self.notify('scrollable')
|
|
|
|
def _adjustment_changed_cb(self, adjustment):
|
|
if adjustment.value <= adjustment.lower:
|
|
can_scroll_prev = False
|
|
else:
|
|
can_scroll_prev = True
|
|
|
|
if adjustment.value + adjustment.page_size >= adjustment.upper:
|
|
can_scroll_next = False
|
|
else:
|
|
can_scroll_next = True
|
|
|
|
if can_scroll_prev != self._can_scroll_prev:
|
|
self._can_scroll_prev = can_scroll_prev
|
|
self.notify('can-scroll-prev')
|
|
|
|
if can_scroll_next != self._can_scroll_next:
|
|
self._can_scroll_next = can_scroll_next
|
|
self.notify('can-scroll-next')
|
|
|
|
|
|
class _TrayScrollButton(ToolButton):
|
|
def __init__(self, icon_name, scroll_direction):
|
|
ToolButton.__init__(self)
|
|
self._viewport = None
|
|
|
|
self._scroll_direction = scroll_direction
|
|
|
|
self.set_size_request(style.GRID_CELL_SIZE, style.GRID_CELL_SIZE)
|
|
|
|
self.icon = Icon(icon_name = icon_name,
|
|
icon_size=gtk.ICON_SIZE_SMALL_TOOLBAR)
|
|
# The alignment is a hack to work around gtk.ToolButton code
|
|
# that sets the icon_size when the icon_widget is a gtk.Image
|
|
alignment = gtk.Alignment(0.5, 0.5)
|
|
alignment.add(self.icon)
|
|
self.set_icon_widget(alignment)
|
|
alignment.show_all()
|
|
|
|
self.connect('clicked', self._clicked_cb)
|
|
|
|
def set_viewport(self, viewport):
|
|
self._viewport = viewport
|
|
self._viewport.connect('notify::scrollable',
|
|
self._viewport_scrollable_changed_cb)
|
|
|
|
if self._scroll_direction == _PREVIOUS_PAGE:
|
|
self._viewport.connect('notify::can-scroll-prev',
|
|
self._viewport_can_scroll_dir_changed_cb)
|
|
self.set_sensitive(self._viewport.props.can_scroll_prev)
|
|
else:
|
|
self._viewport.connect('notify::can-scroll-next',
|
|
self._viewport_can_scroll_dir_changed_cb)
|
|
self.set_sensitive(self._viewport.props.can_scroll_next)
|
|
|
|
|
|
def _viewport_scrollable_changed_cb(self, viewport, pspec):
|
|
self.props.visible = self._viewport.props.scrollable
|
|
|
|
def _viewport_can_scroll_dir_changed_cb(self, viewport, pspec):
|
|
if self._scroll_direction == _PREVIOUS_PAGE:
|
|
sensitive = self._viewport.props.can_scroll_prev
|
|
else:
|
|
sensitive = self._viewport.props.can_scroll_next
|
|
|
|
self.set_sensitive(sensitive)
|
|
|
|
def _clicked_cb(self, button):
|
|
self._viewport.scroll(self._scroll_direction)
|
|
|
|
viewport = property(fset=set_viewport)
|
|
|
|
ALIGN_TO_START = 0
|
|
ALIGN_TO_END = 1
|
|
|
|
class HTray(gtk.HBox):
|
|
__gtype_name__ = 'SugarHTray'
|
|
|
|
__gproperties__ = {
|
|
'align' : (int, None, None, 0, 1, ALIGN_TO_START,
|
|
gobject.PARAM_READWRITE |
|
|
gobject.PARAM_CONSTRUCT_ONLY),
|
|
'drag-active' : (bool, None, None, False,
|
|
gobject.PARAM_READWRITE)
|
|
}
|
|
def __init__(self, **kwargs):
|
|
self._drag_active = False
|
|
self.align = ALIGN_TO_START
|
|
|
|
gobject.GObject.__init__(self, **kwargs)
|
|
|
|
scroll_left = _TrayScrollButton('go-left', _PREVIOUS_PAGE)
|
|
self.pack_start(scroll_left, False)
|
|
|
|
self._viewport = _TrayViewport(gtk.ORIENTATION_HORIZONTAL)
|
|
self.pack_start(self._viewport)
|
|
self._viewport.show()
|
|
|
|
scroll_right = _TrayScrollButton('go-right', _NEXT_PAGE)
|
|
self.pack_start(scroll_right, False)
|
|
|
|
scroll_left.viewport = self._viewport
|
|
scroll_right.viewport = self._viewport
|
|
|
|
if self.align == ALIGN_TO_END:
|
|
spacer = gtk.SeparatorToolItem()
|
|
spacer.set_size_request(0, 0)
|
|
spacer.props.draw = False
|
|
spacer.set_expand(True)
|
|
self._viewport.traybar.insert(spacer, 0)
|
|
spacer.show()
|
|
|
|
def do_set_property(self, pspec, value):
|
|
if pspec.name == 'align':
|
|
self.align = value
|
|
elif pspec.name == 'drag-active':
|
|
self._set_drag_active(value)
|
|
else:
|
|
raise AssertionError
|
|
|
|
def do_get_property(self, pspec):
|
|
if pspec.name == 'align':
|
|
return self.align
|
|
elif pspec.name == 'drag-active':
|
|
return self._drag_active
|
|
else:
|
|
raise AssertionError
|
|
|
|
def _set_drag_active(self, active):
|
|
if self._drag_active != active:
|
|
self._drag_active = active
|
|
if self._drag_active:
|
|
self._viewport.traybar.modify_bg(gtk.STATE_NORMAL,
|
|
style.COLOR_BLACK.get_gdk_color())
|
|
else:
|
|
self._viewport.traybar.modify_bg(gtk.STATE_NORMAL, None)
|
|
|
|
def get_children(self):
|
|
children = self._viewport.traybar.get_children()[:]
|
|
if self.align == ALIGN_TO_END:
|
|
children = children[1:]
|
|
return children
|
|
|
|
def add_item(self, item, index=-1):
|
|
if self.align == ALIGN_TO_END and index > -1:
|
|
index += 1
|
|
self._viewport.traybar.insert(item, index)
|
|
|
|
def remove_item(self, item):
|
|
self._viewport.traybar.remove(item)
|
|
|
|
def get_item_index(self, item):
|
|
index = self._viewport.traybar.get_item_index(item)
|
|
if self.align == ALIGN_TO_END:
|
|
index -= 1
|
|
return index
|
|
|
|
def scroll_to_item(self, item):
|
|
self._viewport.scroll_to_item(item)
|
|
|
|
class VTray(gtk.VBox):
|
|
__gtype_name__ = 'SugarVTray'
|
|
|
|
__gproperties__ = {
|
|
'align' : (int, None, None, 0, 1, ALIGN_TO_START,
|
|
gobject.PARAM_READWRITE |
|
|
gobject.PARAM_CONSTRUCT_ONLY),
|
|
'drag-active' : (bool, None, None, False,
|
|
gobject.PARAM_READWRITE)
|
|
}
|
|
|
|
def __init__(self, **kwargs):
|
|
self._drag_active = False
|
|
self.align = ALIGN_TO_START
|
|
|
|
gobject.GObject.__init__(self, **kwargs)
|
|
|
|
scroll_up = _TrayScrollButton('go-up', _PREVIOUS_PAGE)
|
|
self.pack_start(scroll_up, False)
|
|
|
|
self._viewport = _TrayViewport(gtk.ORIENTATION_VERTICAL)
|
|
self.pack_start(self._viewport)
|
|
self._viewport.show()
|
|
|
|
scroll_down = _TrayScrollButton('go-down', _NEXT_PAGE)
|
|
self.pack_start(scroll_down, False)
|
|
|
|
scroll_up.viewport = self._viewport
|
|
scroll_down.viewport = self._viewport
|
|
|
|
if self.align == ALIGN_TO_END:
|
|
spacer = gtk.SeparatorToolItem()
|
|
spacer.set_size_request(0, 0)
|
|
spacer.props.draw = False
|
|
spacer.set_expand(True)
|
|
self._viewport.traybar.insert(spacer, 0)
|
|
spacer.show()
|
|
|
|
def do_set_property(self, pspec, value):
|
|
if pspec.name == 'align':
|
|
self.align = value
|
|
elif pspec.name == 'drag-active':
|
|
self._set_drag_active(value)
|
|
else:
|
|
raise AssertionError
|
|
|
|
def do_get_property(self, pspec):
|
|
if pspec.name == 'align':
|
|
return self.align
|
|
elif pspec.name == 'drag-active':
|
|
return self._drag_active
|
|
else:
|
|
raise AssertionError
|
|
|
|
def _set_drag_active(self, active):
|
|
if self._drag_active != active:
|
|
self._drag_active = active
|
|
if self._drag_active:
|
|
self._viewport.traybar.modify_bg(gtk.STATE_NORMAL,
|
|
style.COLOR_BLACK.get_gdk_color())
|
|
else:
|
|
self._viewport.traybar.modify_bg(gtk.STATE_NORMAL, None)
|
|
|
|
def get_children(self):
|
|
children = self._viewport.traybar.get_children()[:]
|
|
if self.align == ALIGN_TO_END:
|
|
children = children[1:]
|
|
return children
|
|
|
|
def add_item(self, item, index=-1):
|
|
if self.align == ALIGN_TO_END and index > -1:
|
|
index += 1
|
|
self._viewport.traybar.insert(item, index)
|
|
|
|
def remove_item(self, item):
|
|
self._viewport.traybar.remove(item)
|
|
|
|
def get_item_index(self, item):
|
|
index = self._viewport.traybar.get_item_index(item)
|
|
if self.align == ALIGN_TO_END:
|
|
index -= 1
|
|
return index
|
|
|
|
def scroll_to_item(self, item):
|
|
self._viewport.scroll_to_item(item)
|
|
|
|
class TrayButton(ToolButton):
|
|
def __init__(self, **kwargs):
|
|
ToolButton.__init__(self, **kwargs)
|
|
|
|
class _IconWidget(gtk.EventBox):
|
|
__gtype_name__ = "SugarTrayIconWidget"
|
|
|
|
def __init__(self, icon_name=None, xo_color=None):
|
|
gtk.EventBox.__init__(self)
|
|
|
|
self.set_app_paintable(True)
|
|
|
|
self._icon = Icon(icon_name=icon_name, xo_color=xo_color,
|
|
icon_size=gtk.ICON_SIZE_LARGE_TOOLBAR)
|
|
self.add(self._icon)
|
|
self._icon.show()
|
|
|
|
def do_expose_event(self, event):
|
|
palette = self.parent.palette
|
|
if palette and palette.is_up():
|
|
invoker = palette.props.invoker
|
|
invoker.draw_rectangle(event, palette)
|
|
|
|
gtk.EventBox.do_expose_event(self, event)
|
|
|
|
def get_icon(self):
|
|
return self._icon
|
|
|
|
class TrayIcon(gtk.ToolItem):
|
|
__gtype_name__ = "SugarTrayIcon"
|
|
|
|
def __init__(self, icon_name=None, xo_color=None):
|
|
gtk.ToolItem.__init__(self)
|
|
|
|
self._icon_widget = _IconWidget(icon_name, xo_color)
|
|
self.add(self._icon_widget)
|
|
self._icon_widget.show()
|
|
|
|
self._palette_invoker = ToolInvoker(self)
|
|
|
|
self.set_size_request(style.GRID_CELL_SIZE, style.GRID_CELL_SIZE)
|
|
|
|
self.connect('destroy', self.__destroy_cb)
|
|
|
|
def __destroy_cb(self, icon):
|
|
if self._palette_invoker is not None:
|
|
self._palette_invoker.detach()
|
|
|
|
def create_palette(self):
|
|
return None
|
|
|
|
def get_palette(self):
|
|
return self._palette_invoker.palette
|
|
|
|
def set_palette(self, palette):
|
|
self._palette_invoker.palette = palette
|
|
|
|
palette = gobject.property(
|
|
type=object, setter=set_palette, getter=get_palette)
|
|
|
|
def get_palette_invoker(self):
|
|
return self._palette_invoker
|
|
|
|
def set_palette_invoker(self, palette_invoker):
|
|
self._palette_invoker.detach()
|
|
self._palette_invoker = palette_invoker
|
|
|
|
palette_invoker = gobject.property(
|
|
type=object, setter=set_palette_invoker, getter=get_palette_invoker)
|
|
|
|
def get_icon(self):
|
|
return self._icon_widget.get_icon()
|
|
icon = property(get_icon, None)
|
|
|