Split up the browser, some cleanups and
This commit is contained in:
parent
b4ebdfa5af
commit
f96fbfc10b
@ -11,7 +11,9 @@ TODO="C0301,W0201,R0201"
|
|||||||
#
|
#
|
||||||
# W0613 Unused argument (glib signals)
|
# W0613 Unused argument (glib signals)
|
||||||
# W0511 - FIXME/TODO/XXX
|
# W0511 - FIXME/TODO/XXX
|
||||||
DISABLE="W0613,W0511"
|
# W0302 - Too many lines in module
|
||||||
|
# I0011 - Locally disabled messages
|
||||||
|
DISABLE="W0613,W0511,W0612,W0302,I0011"
|
||||||
|
|
||||||
MSGS="$TODO,$DISABLE"
|
MSGS="$TODO,$DISABLE"
|
||||||
ARG="sugar"
|
ARG="sugar"
|
||||||
|
68
sugar/browser/AddressItem.py
Normal file
68
sugar/browser/AddressItem.py
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
import pygtk
|
||||||
|
pygtk.require('2.0')
|
||||||
|
import gtk
|
||||||
|
|
||||||
|
import sugar.env
|
||||||
|
|
||||||
|
class AddressEntry(gtk.HBox):
|
||||||
|
def __init__(self, callback):
|
||||||
|
gtk.HBox.__init__(self)
|
||||||
|
|
||||||
|
self.callback = callback
|
||||||
|
self.folded = True
|
||||||
|
|
||||||
|
label = gtk.Label("Open")
|
||||||
|
self.pack_start(label, False)
|
||||||
|
label.show()
|
||||||
|
|
||||||
|
self.button = gtk.Button()
|
||||||
|
self.button.set_relief(gtk.RELIEF_NONE)
|
||||||
|
self.button.connect("clicked", self.__button_clicked_cb)
|
||||||
|
self.pack_start(self.button, False)
|
||||||
|
self.button.show()
|
||||||
|
|
||||||
|
self.entry = gtk.Entry()
|
||||||
|
self.entry.connect("activate", self.__activate_cb)
|
||||||
|
self.pack_start(self.entry, False)
|
||||||
|
self.entry.show()
|
||||||
|
|
||||||
|
self._update_folded_state()
|
||||||
|
|
||||||
|
def _update_folded_state(self):
|
||||||
|
if self.folded:
|
||||||
|
image = gtk.Image()
|
||||||
|
image.set_from_file(sugar.env.get_data_file('unfold.png'))
|
||||||
|
self.button.set_image(image)
|
||||||
|
image.show()
|
||||||
|
|
||||||
|
self.entry.hide()
|
||||||
|
else:
|
||||||
|
image = gtk.Image()
|
||||||
|
image.set_from_file(sugar.env.get_data_file('fold.png'))
|
||||||
|
self.button.set_image(image)
|
||||||
|
image.show()
|
||||||
|
|
||||||
|
self.entry.show()
|
||||||
|
self.entry.grab_focus()
|
||||||
|
|
||||||
|
def get_folded(self):
|
||||||
|
return self.folded
|
||||||
|
|
||||||
|
def set_folded(self, folded):
|
||||||
|
self.folded = folded
|
||||||
|
self._update_folded_state()
|
||||||
|
|
||||||
|
def __button_clicked_cb(self, button):
|
||||||
|
self.set_folded(not self.get_folded())
|
||||||
|
|
||||||
|
def __activate_cb(self, entry):
|
||||||
|
self.callback(entry.get_text())
|
||||||
|
self.set_folded(True)
|
||||||
|
|
||||||
|
class AddressItem(gtk.ToolItem):
|
||||||
|
def __init__(self, callback):
|
||||||
|
gtk.ToolItem.__init__(self)
|
||||||
|
|
||||||
|
address_entry = AddressEntry(callback)
|
||||||
|
self.add(address_entry)
|
||||||
|
address_entry.show()
|
124
sugar/browser/BrowserActivity.py
Normal file
124
sugar/browser/BrowserActivity.py
Normal file
@ -0,0 +1,124 @@
|
|||||||
|
from xml.sax import saxutils
|
||||||
|
|
||||||
|
import dbus
|
||||||
|
import pygtk
|
||||||
|
pygtk.require('2.0')
|
||||||
|
import gtk
|
||||||
|
import geckoembed
|
||||||
|
|
||||||
|
from sugar.shell import activity
|
||||||
|
from sugar.browser import NotificationBar
|
||||||
|
from sugar.browser import NavigationToolbar
|
||||||
|
|
||||||
|
class BrowserActivity(activity.Activity):
|
||||||
|
SOLO = 1
|
||||||
|
FOLLOWING = 2
|
||||||
|
LEADING = 3
|
||||||
|
|
||||||
|
def __init__(self, group, uri):
|
||||||
|
activity.Activity.__init__(self)
|
||||||
|
|
||||||
|
self.uri = uri
|
||||||
|
self._group = group
|
||||||
|
self._mode = BrowserActivity.SOLO
|
||||||
|
|
||||||
|
def _update_shared_location(self):
|
||||||
|
address = self.embed.get_address()
|
||||||
|
self._model.set_value('address', address)
|
||||||
|
title = self.embed.get_title()
|
||||||
|
self._model.set_value('title', title)
|
||||||
|
|
||||||
|
def __notif_bar_action_cb(self, bar, action_id):
|
||||||
|
print action_id
|
||||||
|
if action_id == 'set_shared_location':
|
||||||
|
self._update_shared_location()
|
||||||
|
elif action_id == 'goto_shared_location':
|
||||||
|
address = self._model.get_value("address")
|
||||||
|
print address
|
||||||
|
self.embed.load_address(address)
|
||||||
|
self._notif_bar.hide()
|
||||||
|
|
||||||
|
def set_mode(self, mode):
|
||||||
|
self._mode = mode
|
||||||
|
if mode == BrowserActivity.LEADING:
|
||||||
|
self._notif_bar.set_text('Share this page with the group.')
|
||||||
|
self._notif_bar.set_action('set_shared_location', 'Share')
|
||||||
|
self._notif_bar.set_icon('stock_shared-by-me')
|
||||||
|
self._notif_bar.show()
|
||||||
|
|
||||||
|
def _setup_shared(self, uri):
|
||||||
|
self._model = self._group.get_store().get_model(uri)
|
||||||
|
if self._model:
|
||||||
|
self.set_mode(BrowserActivity.FOLLOWING)
|
||||||
|
self._model.add_listener(self.__shared_location_changed_cb)
|
||||||
|
|
||||||
|
def activity_on_connected_to_shell(self):
|
||||||
|
self.activity_set_ellipsize_tab(True)
|
||||||
|
self.activity_set_can_close(True)
|
||||||
|
self.activity_set_tab_text("Web Page")
|
||||||
|
self.activity_set_tab_icon_name("web-browser")
|
||||||
|
self.activity_show_icon(True)
|
||||||
|
|
||||||
|
vbox = gtk.VBox()
|
||||||
|
|
||||||
|
self._notif_bar = NotificationBar()
|
||||||
|
vbox.pack_start(self._notif_bar, False)
|
||||||
|
self._notif_bar.connect('action', self.__notif_bar_action_cb)
|
||||||
|
|
||||||
|
self.embed = geckoembed.Embed()
|
||||||
|
self.embed.connect("title", self.__title_cb)
|
||||||
|
vbox.pack_start(self.embed)
|
||||||
|
|
||||||
|
self.embed.show()
|
||||||
|
self.embed.load_address(self.uri)
|
||||||
|
|
||||||
|
nav_toolbar = NavigationToolbar(self)
|
||||||
|
vbox.pack_start(nav_toolbar, False)
|
||||||
|
nav_toolbar.show()
|
||||||
|
|
||||||
|
plug = self.activity_get_gtk_plug()
|
||||||
|
plug.add(vbox)
|
||||||
|
plug.show()
|
||||||
|
|
||||||
|
vbox.show()
|
||||||
|
|
||||||
|
self._setup_shared(self.uri)
|
||||||
|
|
||||||
|
def get_embed(self):
|
||||||
|
return self.embed
|
||||||
|
|
||||||
|
def share(self):
|
||||||
|
address = self.embed.get_address()
|
||||||
|
self._model = self._group.get_store().create_model(address)
|
||||||
|
self._model.set_value('owner', self._group.get_owner().get_nick_name())
|
||||||
|
self._update_shared_location()
|
||||||
|
self.set_mode(BrowserActivity.LEADING)
|
||||||
|
|
||||||
|
bus = dbus.SessionBus()
|
||||||
|
proxy_obj = bus.get_object('com.redhat.Sugar.Chat', '/com/redhat/Sugar/Chat')
|
||||||
|
chat_shell = dbus.Interface(proxy_obj, 'com.redhat.Sugar.ChatShell')
|
||||||
|
|
||||||
|
escaped_title = saxutils.escape(self.embed.get_title())
|
||||||
|
escaped_address = saxutils.escape(address)
|
||||||
|
chat_shell.send_text_message('<richtext><link href="' + escaped_address +
|
||||||
|
'">' + escaped_title + '</link></richtext>')
|
||||||
|
|
||||||
|
def __title_cb(self, embed):
|
||||||
|
self.activity_set_tab_text(embed.get_title())
|
||||||
|
|
||||||
|
def __shared_location_changed_cb(self, model, key):
|
||||||
|
self.activity_set_has_changes(True)
|
||||||
|
self._notify_shared_location_change()
|
||||||
|
|
||||||
|
def _notify_shared_location_change(self):
|
||||||
|
owner = self._model.get_value('owner')
|
||||||
|
title = self._model.get_value('title')
|
||||||
|
|
||||||
|
text = '<b>' + owner + '</b> is reading <i>' + title + '</i>'
|
||||||
|
self._notif_bar.set_text(text)
|
||||||
|
self._notif_bar.set_action('goto_shared_location', 'Go There')
|
||||||
|
self._notif_bar.set_icon('stock_right')
|
||||||
|
self._notif_bar.show()
|
||||||
|
|
||||||
|
def activity_on_close_from_user(self):
|
||||||
|
self.activity_shutdown()
|
51
sugar/browser/BrowserShell.py
Normal file
51
sugar/browser/BrowserShell.py
Normal file
@ -0,0 +1,51 @@
|
|||||||
|
import dbus
|
||||||
|
import geckoembed
|
||||||
|
|
||||||
|
import sugar.env
|
||||||
|
|
||||||
|
from sugar.p2p.Group import LocalGroup
|
||||||
|
from sugar.browser.WebActivity import WebActivity
|
||||||
|
from sugar.browser.BrowserActivity import BrowserActivity
|
||||||
|
|
||||||
|
class BrowserShell(dbus.service.Object):
|
||||||
|
instance = None
|
||||||
|
|
||||||
|
def get_instance():
|
||||||
|
if not BrowserShell.instance:
|
||||||
|
BrowserShell.instance = BrowserShell()
|
||||||
|
return BrowserShell.instance
|
||||||
|
|
||||||
|
get_instance = staticmethod(get_instance)
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
geckoembed.set_profile_path(sugar.env.get_user_dir())
|
||||||
|
|
||||||
|
session_bus = dbus.SessionBus()
|
||||||
|
bus_name = dbus.service.BusName('com.redhat.Sugar.Browser', bus=session_bus)
|
||||||
|
object_path = '/com/redhat/Sugar/Browser'
|
||||||
|
|
||||||
|
dbus.service.Object.__init__(self, bus_name, object_path)
|
||||||
|
|
||||||
|
self.__browsers = []
|
||||||
|
self._group = LocalGroup()
|
||||||
|
|
||||||
|
def open_web_activity(self):
|
||||||
|
web_activity = WebActivity(self)
|
||||||
|
web_activity.activity_connect_to_shell()
|
||||||
|
|
||||||
|
@dbus.service.method('com.redhat.Sugar.BrowserShell')
|
||||||
|
def get_links(self):
|
||||||
|
links = []
|
||||||
|
for browser in self.__browsers:
|
||||||
|
embed = browser.get_embed()
|
||||||
|
link = {}
|
||||||
|
link['title'] = embed.get_title()
|
||||||
|
link['address'] = embed.get_address()
|
||||||
|
links.append(link)
|
||||||
|
return links
|
||||||
|
|
||||||
|
@dbus.service.method('com.redhat.Sugar.BrowserShell')
|
||||||
|
def open_browser(self, uri):
|
||||||
|
browser = BrowserActivity(self._group, uri)
|
||||||
|
self.__browsers.append(browser)
|
||||||
|
browser.activity_connect_to_shell()
|
73
sugar/browser/NavigationToolbar.py
Normal file
73
sugar/browser/NavigationToolbar.py
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
import pygtk
|
||||||
|
pygtk.require('2.0')
|
||||||
|
import gtk
|
||||||
|
|
||||||
|
from sugar.browser.AddressItem import AddressItem
|
||||||
|
|
||||||
|
class NavigationToolbar(gtk.Toolbar):
|
||||||
|
def __init__(self, browser):
|
||||||
|
gtk.Toolbar.__init__(self)
|
||||||
|
self._browser = browser
|
||||||
|
self._embed = self._browser.get_embed()
|
||||||
|
|
||||||
|
self.set_style(gtk.TOOLBAR_BOTH_HORIZ)
|
||||||
|
|
||||||
|
self.back = gtk.ToolButton(gtk.STOCK_GO_BACK)
|
||||||
|
self.back.connect("clicked", self.__go_back_cb)
|
||||||
|
self.insert(self.back, -1)
|
||||||
|
self.back.show()
|
||||||
|
|
||||||
|
self.forward = gtk.ToolButton(gtk.STOCK_GO_FORWARD)
|
||||||
|
self.forward.connect("clicked", self.__go_forward_cb)
|
||||||
|
self.insert(self.forward, -1)
|
||||||
|
self.forward.show()
|
||||||
|
|
||||||
|
self.reload = gtk.ToolButton(gtk.STOCK_REFRESH)
|
||||||
|
self.reload.connect("clicked", self.__reload_cb)
|
||||||
|
self.insert(self.reload, -1)
|
||||||
|
self.reload.show()
|
||||||
|
|
||||||
|
separator = gtk.SeparatorToolItem()
|
||||||
|
self.insert(separator, -1)
|
||||||
|
separator.show()
|
||||||
|
|
||||||
|
share = gtk.ToolButton(None, "Share")
|
||||||
|
share.set_icon_name('stock_shared-by-me')
|
||||||
|
share.set_is_important(True)
|
||||||
|
share.connect("clicked", self.__share_cb)
|
||||||
|
self.insert(share, -1)
|
||||||
|
share.show()
|
||||||
|
|
||||||
|
separator = gtk.SeparatorToolItem()
|
||||||
|
self.insert(separator, -1)
|
||||||
|
separator.show()
|
||||||
|
|
||||||
|
address_item = AddressItem(self.__open_address_cb)
|
||||||
|
self.insert(address_item, -1)
|
||||||
|
address_item.show()
|
||||||
|
|
||||||
|
self._update_sensitivity()
|
||||||
|
|
||||||
|
self._embed.connect("location", self.__location_changed)
|
||||||
|
|
||||||
|
def _update_sensitivity(self):
|
||||||
|
self.back.set_sensitive(self._embed.can_go_back())
|
||||||
|
self.forward.set_sensitive(self._embed.can_go_forward())
|
||||||
|
|
||||||
|
def __go_back_cb(self, button):
|
||||||
|
self._embed.go_back()
|
||||||
|
|
||||||
|
def __go_forward_cb(self, button):
|
||||||
|
self._embed.go_forward()
|
||||||
|
|
||||||
|
def __reload_cb(self, button):
|
||||||
|
self._embed.reload()
|
||||||
|
|
||||||
|
def __share_cb(self, button):
|
||||||
|
self._browser.share()
|
||||||
|
|
||||||
|
def __location_changed(self, embed):
|
||||||
|
self._update_sensitivity()
|
||||||
|
|
||||||
|
def __open_address_cb(self, address):
|
||||||
|
self._embed.load_address(address)
|
@ -2,7 +2,6 @@ import pygtk
|
|||||||
pygtk.require('2.0')
|
pygtk.require('2.0')
|
||||||
import gtk
|
import gtk
|
||||||
import gobject
|
import gobject
|
||||||
import cairo
|
|
||||||
|
|
||||||
class NotificationBar(gtk.HBox):
|
class NotificationBar(gtk.HBox):
|
||||||
__gsignals__ = {
|
__gsignals__ = {
|
||||||
|
59
sugar/browser/WebActivity.py
Normal file
59
sugar/browser/WebActivity.py
Normal file
@ -0,0 +1,59 @@
|
|||||||
|
import pygtk
|
||||||
|
pygtk.require('2.0')
|
||||||
|
import gtk
|
||||||
|
import geckoembed
|
||||||
|
|
||||||
|
from sugar.shell import activity
|
||||||
|
from sugar.browser.AddressItem import AddressItem
|
||||||
|
|
||||||
|
class AddressToolbar(gtk.Toolbar):
|
||||||
|
def __init__(self, shell):
|
||||||
|
gtk.Toolbar.__init__(self)
|
||||||
|
|
||||||
|
self._shell = shell
|
||||||
|
|
||||||
|
address_item = AddressItem(self.__open_address_cb)
|
||||||
|
self.insert(address_item, 0)
|
||||||
|
address_item.show()
|
||||||
|
|
||||||
|
def __open_address_cb(self, address):
|
||||||
|
self._shell.open_browser(address)
|
||||||
|
|
||||||
|
class WebActivity(activity.Activity):
|
||||||
|
def __init__(self, shell):
|
||||||
|
activity.Activity.__init__(self)
|
||||||
|
self._shell = shell
|
||||||
|
|
||||||
|
def activity_on_connected_to_shell(self):
|
||||||
|
self.activity_set_tab_text("Web")
|
||||||
|
self.activity_set_tab_icon_name("web-browser")
|
||||||
|
self.activity_show_icon(True)
|
||||||
|
|
||||||
|
vbox = gtk.VBox()
|
||||||
|
|
||||||
|
self.embed = geckoembed.Embed()
|
||||||
|
self.embed.connect("open-address", self.__open_address);
|
||||||
|
vbox.pack_start(self.embed)
|
||||||
|
self.embed.show()
|
||||||
|
|
||||||
|
address_toolbar = AddressToolbar(self._shell)
|
||||||
|
vbox.pack_start(address_toolbar, False)
|
||||||
|
address_toolbar.show()
|
||||||
|
|
||||||
|
plug = self.activity_get_gtk_plug()
|
||||||
|
plug.add(vbox)
|
||||||
|
plug.show()
|
||||||
|
|
||||||
|
vbox.show()
|
||||||
|
|
||||||
|
self.embed.load_address("http://www.google.com")
|
||||||
|
|
||||||
|
def __open_address(self, embed, uri, data=None):
|
||||||
|
if uri.startswith("http://www.google.com"):
|
||||||
|
return False
|
||||||
|
else:
|
||||||
|
self._shell.open_browser(uri)
|
||||||
|
return True
|
||||||
|
|
||||||
|
def activity_on_disconnected_from_shell(self):
|
||||||
|
gtk.main_quit()
|
@ -1,359 +1,13 @@
|
|||||||
#!/usr/bin/env python
|
#!/usr/bin/env python
|
||||||
import sys
|
|
||||||
|
|
||||||
from xml.sax import saxutils
|
|
||||||
|
|
||||||
import dbus
|
|
||||||
import dbus.service
|
|
||||||
import dbus.glib
|
|
||||||
|
|
||||||
import pygtk
|
import pygtk
|
||||||
pygtk.require('2.0')
|
pygtk.require('2.0')
|
||||||
import gtk
|
import gtk
|
||||||
|
|
||||||
import geckoembed
|
|
||||||
|
|
||||||
from sugar.session.LogWriter import LogWriter
|
|
||||||
from sugar.shell import activity
|
|
||||||
from sugar.p2p.Group import LocalGroup
|
|
||||||
from sugar.browser.NotificationBar import NotificationBar
|
|
||||||
import sugar.env
|
import sugar.env
|
||||||
|
|
||||||
class AddressToolbar(gtk.Toolbar):
|
from sugar.session.LogWriter import LogWriter
|
||||||
def __init__(self):
|
from sugar.browser.BrowserShell import BrowserShell
|
||||||
gtk.Toolbar.__init__(self)
|
|
||||||
|
|
||||||
address_item = AddressItem(self.__open_address_cb)
|
|
||||||
self.insert(address_item, 0)
|
|
||||||
address_item.show()
|
|
||||||
|
|
||||||
def __open_address_cb(self, address):
|
|
||||||
BrowserShell.get_instance().open_browser(address)
|
|
||||||
|
|
||||||
class AddressItem(gtk.ToolItem):
|
|
||||||
def __init__(self, callback):
|
|
||||||
gtk.ToolItem.__init__(self)
|
|
||||||
|
|
||||||
address_entry = AddressEntry(callback)
|
|
||||||
self.add(address_entry)
|
|
||||||
address_entry.show()
|
|
||||||
|
|
||||||
class AddressEntry(gtk.HBox):
|
|
||||||
def __init__(self, callback):
|
|
||||||
gtk.HBox.__init__(self)
|
|
||||||
|
|
||||||
self.callback = callback
|
|
||||||
self.folded = True
|
|
||||||
|
|
||||||
label = gtk.Label("Open")
|
|
||||||
self.pack_start(label, False)
|
|
||||||
label.show()
|
|
||||||
|
|
||||||
self.button = gtk.Button()
|
|
||||||
self.button.set_relief(gtk.RELIEF_NONE)
|
|
||||||
self.button.connect("clicked", self.__button_clicked_cb)
|
|
||||||
self.pack_start(self.button, False)
|
|
||||||
self.button.show()
|
|
||||||
|
|
||||||
self.entry = gtk.Entry()
|
|
||||||
self.entry.connect("activate", self.__activate_cb)
|
|
||||||
self.pack_start(self.entry, False)
|
|
||||||
self.entry.show()
|
|
||||||
|
|
||||||
self._update_folded_state()
|
|
||||||
|
|
||||||
def _update_folded_state(self):
|
|
||||||
if self.folded:
|
|
||||||
image = gtk.Image()
|
|
||||||
image.set_from_file(sugar.env.get_data_file('unfold.png'))
|
|
||||||
self.button.set_image(image)
|
|
||||||
image.show()
|
|
||||||
|
|
||||||
self.entry.hide()
|
|
||||||
else:
|
|
||||||
image = gtk.Image()
|
|
||||||
image.set_from_file(sugar.env.get_data_file('fold.png'))
|
|
||||||
self.button.set_image(image)
|
|
||||||
image.show()
|
|
||||||
|
|
||||||
self.entry.show()
|
|
||||||
self.entry.grab_focus()
|
|
||||||
|
|
||||||
def get_folded(self):
|
|
||||||
return self.folded
|
|
||||||
|
|
||||||
def set_folded(self, folded):
|
|
||||||
self.folded = folded
|
|
||||||
self._update_folded_state()
|
|
||||||
|
|
||||||
def __button_clicked_cb(self, button):
|
|
||||||
self.set_folded(not self.get_folded())
|
|
||||||
|
|
||||||
def __activate_cb(self, entry):
|
|
||||||
self.callback(entry.get_text())
|
|
||||||
self.set_folded(True)
|
|
||||||
|
|
||||||
class NavigationToolbar(gtk.Toolbar):
|
|
||||||
def __init__(self, browser):
|
|
||||||
gtk.Toolbar.__init__(self)
|
|
||||||
self._browser = browser
|
|
||||||
self._embed = self._browser.get_embed()
|
|
||||||
|
|
||||||
self.set_style(gtk.TOOLBAR_BOTH_HORIZ)
|
|
||||||
|
|
||||||
self.back = gtk.ToolButton(gtk.STOCK_GO_BACK)
|
|
||||||
self.back.connect("clicked", self.__go_back_cb)
|
|
||||||
self.insert(self.back, -1)
|
|
||||||
self.back.show()
|
|
||||||
|
|
||||||
self.forward = gtk.ToolButton(gtk.STOCK_GO_FORWARD)
|
|
||||||
self.forward.connect("clicked", self.__go_forward_cb)
|
|
||||||
self.insert(self.forward, -1)
|
|
||||||
self.forward.show()
|
|
||||||
|
|
||||||
self.reload = gtk.ToolButton(gtk.STOCK_REFRESH)
|
|
||||||
self.reload.connect("clicked", self.__reload_cb)
|
|
||||||
self.insert(self.reload, -1)
|
|
||||||
self.reload.show()
|
|
||||||
|
|
||||||
separator = gtk.SeparatorToolItem()
|
|
||||||
self.insert(separator, -1)
|
|
||||||
separator.show()
|
|
||||||
|
|
||||||
share = gtk.ToolButton(None, "Share")
|
|
||||||
share.set_icon_name('stock_shared-by-me')
|
|
||||||
share.set_is_important(True)
|
|
||||||
share.connect("clicked", self.__share_cb)
|
|
||||||
self.insert(share, -1)
|
|
||||||
share.show()
|
|
||||||
|
|
||||||
separator = gtk.SeparatorToolItem()
|
|
||||||
self.insert(separator, -1)
|
|
||||||
separator.show()
|
|
||||||
|
|
||||||
address_item = AddressItem(self.__open_address_cb)
|
|
||||||
self.insert(address_item, -1)
|
|
||||||
address_item.show()
|
|
||||||
|
|
||||||
self._update_sensitivity()
|
|
||||||
|
|
||||||
self._embed.connect("location", self.__location_changed)
|
|
||||||
|
|
||||||
def _update_sensitivity(self):
|
|
||||||
self.back.set_sensitive(self._embed.can_go_back())
|
|
||||||
self.forward.set_sensitive(self._embed.can_go_forward())
|
|
||||||
|
|
||||||
def __go_back_cb(self, button):
|
|
||||||
self._embed.go_back()
|
|
||||||
|
|
||||||
def __go_forward_cb(self, button):
|
|
||||||
self._embed.go_forward()
|
|
||||||
|
|
||||||
def __reload_cb(self, button):
|
|
||||||
self._embed.reload()
|
|
||||||
|
|
||||||
def __share_cb(self, button):
|
|
||||||
self._browser.share()
|
|
||||||
|
|
||||||
def __location_changed(self, embed):
|
|
||||||
self._update_sensitivity()
|
|
||||||
|
|
||||||
def __open_address_cb(self, address):
|
|
||||||
self._embed.load_address(address)
|
|
||||||
|
|
||||||
class BrowserActivity(activity.Activity):
|
|
||||||
SOLO = 1
|
|
||||||
FOLLOWING = 2
|
|
||||||
LEADING = 3
|
|
||||||
|
|
||||||
def __init__(self, group, uri):
|
|
||||||
activity.Activity.__init__(self)
|
|
||||||
|
|
||||||
self.uri = uri
|
|
||||||
self._group = group
|
|
||||||
self._mode = BrowserActivity.SOLO
|
|
||||||
|
|
||||||
def _update_shared_location(self):
|
|
||||||
address = self.embed.get_address()
|
|
||||||
self._model.set_value('address', address)
|
|
||||||
title = self.embed.get_title()
|
|
||||||
self._model.set_value('title', title)
|
|
||||||
|
|
||||||
def __notif_bar_action_cb(self, bar, action_id):
|
|
||||||
print action_id
|
|
||||||
if action_id == 'set_shared_location':
|
|
||||||
self._update_shared_location()
|
|
||||||
elif action_id == 'goto_shared_location':
|
|
||||||
address = self._model.get_value("address")
|
|
||||||
print address
|
|
||||||
self.embed.load_address(address)
|
|
||||||
self._notif_bar.hide()
|
|
||||||
|
|
||||||
def set_mode(self, mode):
|
|
||||||
self._mode = mode
|
|
||||||
if mode == BrowserActivity.LEADING:
|
|
||||||
self._notif_bar.set_text('Share this page with the group.')
|
|
||||||
self._notif_bar.set_action('set_shared_location', 'Share')
|
|
||||||
self._notif_bar.set_icon('stock_shared-by-me')
|
|
||||||
self._notif_bar.show()
|
|
||||||
|
|
||||||
def _setup_shared(self, uri):
|
|
||||||
self._model = self._group.get_store().get_model(uri)
|
|
||||||
if self._model:
|
|
||||||
self.set_mode(BrowserActivity.FOLLOWING)
|
|
||||||
self._model.add_listener(self.__shared_location_changed_cb)
|
|
||||||
|
|
||||||
def activity_on_connected_to_shell(self):
|
|
||||||
self.activity_set_ellipsize_tab(True)
|
|
||||||
self.activity_set_can_close(True)
|
|
||||||
self.activity_set_tab_text("Web Page")
|
|
||||||
self.activity_set_tab_icon_name("web-browser")
|
|
||||||
self.activity_show_icon(True)
|
|
||||||
|
|
||||||
vbox = gtk.VBox()
|
|
||||||
|
|
||||||
self._notif_bar = NotificationBar()
|
|
||||||
vbox.pack_start(self._notif_bar, False)
|
|
||||||
self._notif_bar.connect('action', self.__notif_bar_action_cb)
|
|
||||||
|
|
||||||
self.embed = geckoembed.Embed()
|
|
||||||
self.embed.connect("title", self.__title_cb)
|
|
||||||
vbox.pack_start(self.embed)
|
|
||||||
|
|
||||||
self.embed.show()
|
|
||||||
self.embed.load_address(self.uri)
|
|
||||||
|
|
||||||
nav_toolbar = NavigationToolbar(self)
|
|
||||||
vbox.pack_start(nav_toolbar, False)
|
|
||||||
nav_toolbar.show()
|
|
||||||
|
|
||||||
plug = self.activity_get_gtk_plug()
|
|
||||||
plug.add(vbox)
|
|
||||||
plug.show()
|
|
||||||
|
|
||||||
vbox.show()
|
|
||||||
|
|
||||||
self._setup_shared(self.uri)
|
|
||||||
|
|
||||||
def get_embed(self):
|
|
||||||
return self.embed
|
|
||||||
|
|
||||||
def share(self):
|
|
||||||
address = self.embed.get_address()
|
|
||||||
self._model = self._group.get_store().create_model(address)
|
|
||||||
self._model.set_value('owner', self._group.get_owner().get_nick_name())
|
|
||||||
self._update_shared_location()
|
|
||||||
self.set_mode(BrowserActivity.LEADING)
|
|
||||||
|
|
||||||
bus = dbus.SessionBus()
|
|
||||||
proxy_obj = bus.get_object('com.redhat.Sugar.Chat', '/com/redhat/Sugar/Chat')
|
|
||||||
chat_shell = dbus.Interface(proxy_obj, 'com.redhat.Sugar.ChatShell')
|
|
||||||
|
|
||||||
escaped_title = saxutils.escape(self.embed.get_title())
|
|
||||||
escaped_address = saxutils.escape(address)
|
|
||||||
chat_shell.send_text_message('<richtext><link href="' + escaped_address +
|
|
||||||
'">' + escaped_title + '</link></richtext>')
|
|
||||||
|
|
||||||
def __title_cb(self, embed):
|
|
||||||
self.activity_set_tab_text(embed.get_title())
|
|
||||||
|
|
||||||
def __shared_location_changed_cb(self, model, key):
|
|
||||||
self.activity_set_has_changes(True)
|
|
||||||
self._notify_shared_location_change()
|
|
||||||
|
|
||||||
def _notify_shared_location_change(self):
|
|
||||||
owner = self._model.get_value('owner')
|
|
||||||
title = self._model.get_value('title')
|
|
||||||
|
|
||||||
text = '<b>' + owner + '</b> is reading <i>' + title + '</i>'
|
|
||||||
self._notif_bar.set_text(text)
|
|
||||||
self._notif_bar.set_action('goto_shared_location', 'Go There')
|
|
||||||
self._notif_bar.set_icon('stock_right')
|
|
||||||
self._notif_bar.show()
|
|
||||||
|
|
||||||
def activity_on_close_from_user(self):
|
|
||||||
self.activity_shutdown()
|
|
||||||
|
|
||||||
class WebActivity(activity.Activity):
|
|
||||||
def __init__(self):
|
|
||||||
activity.Activity.__init__(self)
|
|
||||||
|
|
||||||
def activity_on_connected_to_shell(self):
|
|
||||||
self.activity_set_tab_text("Web")
|
|
||||||
self.activity_set_tab_icon_name("web-browser")
|
|
||||||
self.activity_show_icon(True)
|
|
||||||
|
|
||||||
vbox = gtk.VBox()
|
|
||||||
|
|
||||||
self.embed = geckoembed.Embed()
|
|
||||||
self.embed.connect("open-address", self.__open_address);
|
|
||||||
vbox.pack_start(self.embed)
|
|
||||||
self.embed.show()
|
|
||||||
|
|
||||||
address_toolbar = AddressToolbar()
|
|
||||||
vbox.pack_start(address_toolbar, False)
|
|
||||||
address_toolbar.show()
|
|
||||||
|
|
||||||
plug = self.activity_get_gtk_plug()
|
|
||||||
plug.add(vbox)
|
|
||||||
plug.show()
|
|
||||||
|
|
||||||
vbox.show()
|
|
||||||
|
|
||||||
self.embed.load_address("http://www.google.com")
|
|
||||||
|
|
||||||
def __open_address(self, embed, uri, data=None):
|
|
||||||
if uri.startswith("http://www.google.com"):
|
|
||||||
return False
|
|
||||||
else:
|
|
||||||
BrowserShell.get_instance().open_browser(uri)
|
|
||||||
return True
|
|
||||||
|
|
||||||
def activity_on_disconnected_from_shell(self):
|
|
||||||
gtk.main_quit()
|
|
||||||
|
|
||||||
class BrowserShell(dbus.service.Object):
|
|
||||||
instance = None
|
|
||||||
|
|
||||||
def get_instance():
|
|
||||||
if not BrowserShell.instance:
|
|
||||||
BrowserShell.instance = BrowserShell()
|
|
||||||
return BrowserShell.instance
|
|
||||||
|
|
||||||
get_instance = staticmethod(get_instance)
|
|
||||||
|
|
||||||
def __init__(self):
|
|
||||||
geckoembed.set_profile_path(sugar.env.get_user_dir())
|
|
||||||
|
|
||||||
session_bus = dbus.SessionBus()
|
|
||||||
bus_name = dbus.service.BusName('com.redhat.Sugar.Browser', bus=session_bus)
|
|
||||||
object_path = '/com/redhat/Sugar/Browser'
|
|
||||||
|
|
||||||
dbus.service.Object.__init__(self, bus_name, object_path)
|
|
||||||
|
|
||||||
self.__browsers = []
|
|
||||||
self._group = LocalGroup()
|
|
||||||
|
|
||||||
def open_web_activity(self):
|
|
||||||
web_activity = WebActivity()
|
|
||||||
web_activity.activity_connect_to_shell()
|
|
||||||
|
|
||||||
@dbus.service.method('com.redhat.Sugar.BrowserShell')
|
|
||||||
def get_links(self):
|
|
||||||
links = []
|
|
||||||
for browser in self.__browsers:
|
|
||||||
embed = browser.get_embed()
|
|
||||||
link = {}
|
|
||||||
link['title'] = embed.get_title()
|
|
||||||
link['address'] = embed.get_address()
|
|
||||||
links.append(link)
|
|
||||||
return links
|
|
||||||
|
|
||||||
@dbus.service.method('com.redhat.Sugar.BrowserShell')
|
|
||||||
def open_browser(self, uri):
|
|
||||||
browser = BrowserActivity(self._group, uri)
|
|
||||||
self.__browsers.append(browser)
|
|
||||||
browser.activity_connect_to_shell()
|
|
||||||
|
|
||||||
log_writer = LogWriter("Web")
|
log_writer = LogWriter("Web")
|
||||||
log_writer.start()
|
log_writer.start()
|
||||||
|
@ -1,7 +1,5 @@
|
|||||||
#!/usr/bin/python -t
|
#!/usr/bin/env python
|
||||||
# -*- tab-width: 4; indent-tabs-mode: t -*-
|
|
||||||
|
|
||||||
import sys
|
|
||||||
import base64
|
import base64
|
||||||
import sha
|
import sha
|
||||||
|
|
||||||
@ -28,7 +26,7 @@ import sugar.env
|
|||||||
|
|
||||||
import richtext
|
import richtext
|
||||||
|
|
||||||
PANGO_SCALE=1024 # Where is this defined?
|
PANGO_SCALE = 1024 # Where is this defined?
|
||||||
|
|
||||||
CHAT_SERVICE_TYPE = "_olpc_chat._tcp"
|
CHAT_SERVICE_TYPE = "_olpc_chat._tcp"
|
||||||
CHAT_SERVICE_PORT = 6100
|
CHAT_SERVICE_PORT = 6100
|
||||||
@ -700,7 +698,5 @@ log_writer = LogWriter("Chat")
|
|||||||
log_writer.start()
|
log_writer.start()
|
||||||
|
|
||||||
ChatShell.get_instance().open_group_chat()
|
ChatShell.get_instance().open_group_chat()
|
||||||
try:
|
|
||||||
gtk.main()
|
gtk.main()
|
||||||
except KeyboardInterrupt:
|
|
||||||
pass
|
|
||||||
|
@ -1,7 +1,6 @@
|
|||||||
import pygtk
|
import pygtk
|
||||||
pygtk.require('2.0')
|
pygtk.require('2.0')
|
||||||
import gtk
|
import gtk
|
||||||
import cairo
|
|
||||||
|
|
||||||
from Sketch import Sketch
|
from Sketch import Sketch
|
||||||
|
|
||||||
@ -78,8 +77,8 @@ class SketchPad(gtk.DrawingArea):
|
|||||||
d.setSVG(s)
|
d.setSVG(s)
|
||||||
return d.toXml()
|
return d.toXml()
|
||||||
|
|
||||||
def test_quit(w, sketchpad):
|
def test_quit(w, skpad):
|
||||||
print sketchpad.to_svg()
|
print skpad.to_svg()
|
||||||
gtk.main_quit()
|
gtk.main_quit()
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
@ -97,7 +97,7 @@ class Owner(Buddy):
|
|||||||
try:
|
try:
|
||||||
os.makedirs(user_dir)
|
os.makedirs(user_dir)
|
||||||
except OSError:
|
except OSError:
|
||||||
pass
|
print 'Could not create user directory.'
|
||||||
|
|
||||||
for fname in os.listdir(user_dir):
|
for fname in os.listdir(user_dir):
|
||||||
if not fname.startswith("buddy-icon."):
|
if not fname.startswith("buddy-icon."):
|
||||||
|
@ -1,3 +1,7 @@
|
|||||||
|
# FIXME tests use initialized variables, any better
|
||||||
|
# what to shut up pylint for those?
|
||||||
|
# pylint: disable-msg = W0612
|
||||||
|
|
||||||
import socket
|
import socket
|
||||||
import time
|
import time
|
||||||
import sha
|
import sha
|
||||||
@ -11,9 +15,9 @@ pygtk.require('2.0')
|
|||||||
import gtk, gobject
|
import gtk, gobject
|
||||||
|
|
||||||
|
|
||||||
def _stringify_sha(sha):
|
def _stringify_sha(sha_hash):
|
||||||
print_sha = ""
|
print_sha = ""
|
||||||
for char in sha:
|
for char in sha_hash:
|
||||||
print_sha = print_sha + binascii.b2a_hex(char)
|
print_sha = print_sha + binascii.b2a_hex(char)
|
||||||
return print_sha
|
return print_sha
|
||||||
|
|
||||||
@ -1089,7 +1093,7 @@ franker, more natural, as it were -- she really must exclude me from privileges
|
|||||||
try:
|
try:
|
||||||
gtk.main()
|
gtk.main()
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
pass
|
print 'Ctrl+C pressed, exiting...'
|
||||||
|
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
@ -2,11 +2,6 @@ import xmlrpclib
|
|||||||
import socket
|
import socket
|
||||||
import traceback
|
import traceback
|
||||||
|
|
||||||
import pygtk
|
|
||||||
pygtk.require('2.0')
|
|
||||||
import gobject
|
|
||||||
|
|
||||||
|
|
||||||
import network
|
import network
|
||||||
from MostlyReliablePipe import MostlyReliablePipe
|
from MostlyReliablePipe import MostlyReliablePipe
|
||||||
|
|
||||||
|
@ -1,3 +1,5 @@
|
|||||||
|
# pylint: disable-msg = W0221
|
||||||
|
|
||||||
import socket
|
import socket
|
||||||
import threading
|
import threading
|
||||||
import traceback
|
import traceback
|
||||||
@ -132,6 +134,9 @@ class GlibXMLRPCTransport(xmlrpclib.Transport):
|
|||||||
# @param host Target host.
|
# @param host Target host.
|
||||||
# @return A connection handle.
|
# @return A connection handle.
|
||||||
|
|
||||||
|
def __init__(self):
|
||||||
|
pass
|
||||||
|
|
||||||
def make_connection(self, host):
|
def make_connection(self, host):
|
||||||
"""Use our own connection object so we can get its socket."""
|
"""Use our own connection object so we can get its socket."""
|
||||||
# create a HTTP connection object from a host descriptor
|
# create a HTTP connection object from a host descriptor
|
||||||
@ -224,8 +229,8 @@ class GlibServerProxy(xmlrpclib.ServerProxy):
|
|||||||
|
|
||||||
# get the url
|
# get the url
|
||||||
import urllib
|
import urllib
|
||||||
type, uri = urllib.splittype(uri)
|
urltype, uri = urllib.splittype(uri)
|
||||||
if type not in ("http", "https"):
|
if urltype not in ("http", "https"):
|
||||||
raise IOError, "unsupported XML-RPC protocol"
|
raise IOError, "unsupported XML-RPC protocol"
|
||||||
self._host, self._handler = urllib.splithost(uri)
|
self._host, self._handler = urllib.splithost(uri)
|
||||||
if not self._handler:
|
if not self._handler:
|
||||||
@ -330,7 +335,7 @@ def xmlrpc_test():
|
|||||||
|
|
||||||
|
|
||||||
def main():
|
def main():
|
||||||
import gtk, gobject
|
import gtk
|
||||||
server = GlibXMLRPCServer(("", 8888))
|
server = GlibXMLRPCServer(("", 8888))
|
||||||
inst = Test()
|
inst = Test()
|
||||||
server.register_instance(inst)
|
server.register_instance(inst)
|
||||||
@ -340,7 +345,7 @@ def main():
|
|||||||
try:
|
try:
|
||||||
gtk.main()
|
gtk.main()
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
pass
|
print 'Ctrl+C pressed, exiting...'
|
||||||
print "Done."
|
print "Done."
|
||||||
|
|
||||||
if __name__ == "__main__":
|
if __name__ == "__main__":
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
import os
|
import os
|
||||||
import sys
|
import sys
|
||||||
|
|
||||||
import dbus
|
import dbus
|
||||||
|
|
||||||
class LogWriter:
|
class LogWriter:
|
||||||
@ -16,7 +15,4 @@ class LogWriter:
|
|||||||
sys.stderr = self
|
sys.stderr = self
|
||||||
|
|
||||||
def write(self, s):
|
def write(self, s):
|
||||||
try:
|
self._logger.log(self._application, s)
|
||||||
self._logger.log(self._application, s)
|
|
||||||
except:
|
|
||||||
pass
|
|
||||||
|
@ -1,5 +1,4 @@
|
|||||||
import os
|
import os
|
||||||
import sys
|
|
||||||
from ConfigParser import ConfigParser
|
from ConfigParser import ConfigParser
|
||||||
|
|
||||||
import pygtk
|
import pygtk
|
||||||
@ -39,4 +38,4 @@ def start():
|
|||||||
try:
|
try:
|
||||||
gtk.main()
|
gtk.main()
|
||||||
except KeyboardInterrupt:
|
except KeyboardInterrupt:
|
||||||
pass
|
print 'Ctrl+C pressed, exiting...'
|
||||||
|
@ -1,8 +1,3 @@
|
|||||||
#!/usr/bin/python
|
|
||||||
# -*- tab-width: 4; indent-tabs-mode: t -*-
|
|
||||||
|
|
||||||
import sys
|
|
||||||
|
|
||||||
import dbus
|
import dbus
|
||||||
import dbus.service
|
import dbus.service
|
||||||
import dbus.glib
|
import dbus.glib
|
||||||
@ -11,8 +6,6 @@ pygtk.require('2.0')
|
|||||||
import gtk
|
import gtk
|
||||||
import pango
|
import pango
|
||||||
|
|
||||||
from sugar.session.LogWriter import LogWriter
|
|
||||||
|
|
||||||
activity_counter = 0
|
activity_counter = 0
|
||||||
|
|
||||||
class ActivityHost(dbus.service.Object):
|
class ActivityHost(dbus.service.Object):
|
||||||
@ -329,10 +322,10 @@ class ConsoleLogger(dbus.service.Object):
|
|||||||
def __key_press_event_cb(self, window, event):
|
def __key_press_event_cb(self, window, event):
|
||||||
if event.keyval == gtk.keysyms.d and \
|
if event.keyval == gtk.keysyms.d and \
|
||||||
event.state & gtk.gdk.CONTROL_MASK:
|
event.state & gtk.gdk.CONTROL_MASK:
|
||||||
if self._window.get_property('visible'):
|
if self._window.get_property('visible'):
|
||||||
self._window.hide()
|
self._window.hide()
|
||||||
else:
|
else:
|
||||||
self._window.show()
|
self._window.show()
|
||||||
|
|
||||||
@dbus.service.method('com.redhat.Sugar.Logger')
|
@dbus.service.method('com.redhat.Sugar.Logger')
|
||||||
def log(self, application, message):
|
def log(self, application, message):
|
||||||
|
Loading…
Reference in New Issue
Block a user