increase granularity of mesh device control
This commit is contained in:
parent
e52d6f6ed4
commit
6b6470ebcb
2
NEWS
2
NEWS
@ -1,3 +1,5 @@
|
|||||||
|
* #1260, #2664, #1542, #2985: Rework network UI bits to be more informative and
|
||||||
|
increase granularity of mesh device control
|
||||||
* #2909: Make python activities more tolerant to missing metadata properties. (tomeu)
|
* #2909: Make python activities more tolerant to missing metadata properties. (tomeu)
|
||||||
* #2653: Add audio/wav and audio/x-wav as Audio objects. (tomeu)
|
* #2653: Add audio/wav and audio/x-wav as Audio objects. (tomeu)
|
||||||
* Support moving of data files written to the datastore using standard Activity
|
* Support moving of data files written to the datastore using standard Activity
|
||||||
|
@ -207,6 +207,8 @@ class Device(gobject.GObject):
|
|||||||
gobject.TYPE_NONE, ([])),
|
gobject.TYPE_NONE, ([])),
|
||||||
'state-changed': (gobject.SIGNAL_RUN_FIRST,
|
'state-changed': (gobject.SIGNAL_RUN_FIRST,
|
||||||
gobject.TYPE_NONE, ([])),
|
gobject.TYPE_NONE, ([])),
|
||||||
|
'activation-stage-changed': (gobject.SIGNAL_RUN_FIRST,
|
||||||
|
gobject.TYPE_NONE, ([])),
|
||||||
'network-appeared': (gobject.SIGNAL_RUN_FIRST,
|
'network-appeared': (gobject.SIGNAL_RUN_FIRST,
|
||||||
gobject.TYPE_NONE,
|
gobject.TYPE_NONE,
|
||||||
([gobject.TYPE_PYOBJECT])),
|
([gobject.TYPE_PYOBJECT])),
|
||||||
@ -332,6 +334,15 @@ class Device(gobject.GObject):
|
|||||||
ret.append(net.get_op())
|
ret.append(net.get_op())
|
||||||
return ret
|
return ret
|
||||||
|
|
||||||
|
def get_mesh_step(self):
|
||||||
|
if self._type != DEVICE_TYPE_802_11_MESH_OLPC:
|
||||||
|
raise RuntimeError("Only valid for mesh devices")
|
||||||
|
try:
|
||||||
|
step = self.dev.getMeshStep(timeout=3000)
|
||||||
|
except dbus.DBusException, e:
|
||||||
|
step = 0
|
||||||
|
return step
|
||||||
|
|
||||||
def get_frequency(self):
|
def get_frequency(self):
|
||||||
freq = 0.0
|
freq = 0.0
|
||||||
try:
|
try:
|
||||||
@ -426,6 +437,9 @@ class Device(gobject.GObject):
|
|||||||
if state == self._state:
|
if state == self._state:
|
||||||
return
|
return
|
||||||
|
|
||||||
|
if state == DEVICE_STATE_INACTIVE:
|
||||||
|
self._act_stage = 0
|
||||||
|
|
||||||
self._state = state
|
self._state = state
|
||||||
if self._valid:
|
if self._valid:
|
||||||
self.emit('state-changed')
|
self.emit('state-changed')
|
||||||
@ -437,6 +451,16 @@ class Device(gobject.GObject):
|
|||||||
self.dev.getActiveNetwork(reply_handler=lambda *args: self._get_active_net_cb(state, *args),
|
self.dev.getActiveNetwork(reply_handler=lambda *args: self._get_active_net_cb(state, *args),
|
||||||
error_handler=self._get_active_net_error_cb)
|
error_handler=self._get_active_net_error_cb)
|
||||||
|
|
||||||
|
def set_activation_stage(self, stage):
|
||||||
|
if stage == self._act_stage:
|
||||||
|
return
|
||||||
|
self._act_stage = stage
|
||||||
|
if self._valid:
|
||||||
|
self.emit('activation-stage-changed')
|
||||||
|
|
||||||
|
def get_activation_stage(self):
|
||||||
|
return self._act_stage
|
||||||
|
|
||||||
def get_ssid(self):
|
def get_ssid(self):
|
||||||
if self._active_network and self._active_network.is_valid():
|
if self._active_network and self._active_network.is_valid():
|
||||||
return self._active_network.get_ssid()
|
return self._active_network.get_ssid()
|
||||||
@ -586,26 +610,32 @@ class NMClient(gobject.GObject):
|
|||||||
except dbus.DBusException:
|
except dbus.DBusException:
|
||||||
pass
|
pass
|
||||||
|
|
||||||
def set_active_device(self, device, network=None):
|
def set_active_device(self, device, network=None, mesh_freq=None, mesh_start=None):
|
||||||
ssid = ""
|
ssid = ""
|
||||||
if network:
|
if network:
|
||||||
ssid = network.get_ssid()
|
ssid = network.get_ssid()
|
||||||
try:
|
if device.get_type() == DEVICE_TYPE_802_11_MESH_OLPC:
|
||||||
# NM 0.6.4 and earlier have a bug which returns an
|
if mesh_freq or mesh_start:
|
||||||
# InvalidArguments error if no security information is passed
|
if mesh_freq and not mesh_start:
|
||||||
# for wireless networks
|
self._nm_obj.setActiveDevice(device.get_op(), dbus.Double(mesh_freq))
|
||||||
self._nm_obj.setActiveDevice(device.get_op(), ssid)
|
elif mesh_start and not mesh_freq:
|
||||||
except dbus.DBusException, e:
|
self._nm_obj.setActiveDevice(device.get_op(), dbus.Double(0.0), dbus.UInt32(mesh_start))
|
||||||
if str(e).find("invalid arguments"):
|
else:
|
||||||
pass
|
self._nm_obj.setActiveDevice(device.get_op(), dbus.Double(mesh_freq), dbus.UInt32(mesh_start))
|
||||||
else:
|
else:
|
||||||
raise dbus.DBusException(e)
|
self._nm_obj.setActiveDevice(device.get_op())
|
||||||
|
else:
|
||||||
|
self._nm_obj.setActiveDevice(device.get_op(), ssid)
|
||||||
|
|
||||||
def state_changed_sig_handler(self, new_state):
|
def state_changed_sig_handler(self, new_state):
|
||||||
logging.debug('NM State Changed to %d' % new_state)
|
logging.debug('NM State Changed to %d' % new_state)
|
||||||
|
|
||||||
def device_activation_stage_sig_handler(self, device, stage):
|
def device_activation_stage_sig_handler(self, device, stage):
|
||||||
logging.debug('Device Activation Stage "%s" for device %s' % (NM_DEVICE_STAGE_STRINGS[stage], device))
|
logging.debug('Device Activation Stage "%s" for device %s' % (NM_DEVICE_STAGE_STRINGS[stage], device))
|
||||||
|
if not self._devices.has_key(device):
|
||||||
|
logging.debug('DeviceActivationStage, device %s does not exist' % (device))
|
||||||
|
return
|
||||||
|
self._devices[device].set_activation_stage(stage)
|
||||||
|
|
||||||
def device_activating_sig_handler(self, device):
|
def device_activating_sig_handler(self, device):
|
||||||
logging.debug('DeviceActivating for %s' % (device))
|
logging.debug('DeviceActivating for %s' % (device))
|
||||||
|
@ -25,7 +25,10 @@ class Device(device.Device):
|
|||||||
'strength' : (int, None, None, 0, 100, 0,
|
'strength' : (int, None, None, 0, 100, 0,
|
||||||
gobject.PARAM_READABLE),
|
gobject.PARAM_READABLE),
|
||||||
'state' : (int, None, None, device.STATE_ACTIVATING,
|
'state' : (int, None, None, device.STATE_ACTIVATING,
|
||||||
device.STATE_INACTIVE, 0, gobject.PARAM_READABLE)
|
device.STATE_INACTIVE, 0, gobject.PARAM_READABLE),
|
||||||
|
'activation-stage': (int, None, None, 0, 7, 0, gobject.PARAM_READABLE),
|
||||||
|
'frequency': (float, None, None, 0, 2.72, 0, gobject.PARAM_READABLE),
|
||||||
|
'mesh-step': (int, None, None, 0, 4, 0, gobject.PARAM_READABLE),
|
||||||
}
|
}
|
||||||
|
|
||||||
def __init__(self, nm_device):
|
def __init__(self, nm_device):
|
||||||
@ -36,6 +39,8 @@ class Device(device.Device):
|
|||||||
self._strength_changed_cb)
|
self._strength_changed_cb)
|
||||||
self._nm_device.connect('state-changed',
|
self._nm_device.connect('state-changed',
|
||||||
self._state_changed_cb)
|
self._state_changed_cb)
|
||||||
|
self._nm_device.connect('activation-stage-changed',
|
||||||
|
self._activation_stage_changed_cb)
|
||||||
|
|
||||||
def _strength_changed_cb(self, nm_device):
|
def _strength_changed_cb(self, nm_device):
|
||||||
self.notify('strength')
|
self.notify('strength')
|
||||||
@ -43,15 +48,28 @@ class Device(device.Device):
|
|||||||
def _state_changed_cb(self, nm_device):
|
def _state_changed_cb(self, nm_device):
|
||||||
self.notify('state')
|
self.notify('state')
|
||||||
|
|
||||||
|
def _activation_stage_changed_cb(self, nm_device):
|
||||||
|
self.notify('activation-stage')
|
||||||
|
|
||||||
def do_get_property(self, pspec):
|
def do_get_property(self, pspec):
|
||||||
if pspec.name == 'strength':
|
if pspec.name == 'strength':
|
||||||
return self._nm_device.get_strength()
|
return self._nm_device.get_strength()
|
||||||
elif pspec.name == 'state':
|
elif pspec.name == 'state':
|
||||||
nm_state = self._nm_device.get_state()
|
nm_state = self._nm_device.get_state()
|
||||||
return device._nm_state_to_state[nm_state]
|
return device._nm_state_to_state[nm_state]
|
||||||
|
elif pspec.name == 'activation-stage':
|
||||||
|
return self._nm_device.get_activation_stage()
|
||||||
|
elif pspec.name == 'frequency':
|
||||||
|
return self._nm_device.get_frequency()
|
||||||
|
elif pspec.name == 'mesh-step':
|
||||||
|
return self._nm_device.get_mesh_step()
|
||||||
|
|
||||||
def get_type(self):
|
def get_type(self):
|
||||||
return 'network.mesh'
|
return 'network.mesh'
|
||||||
|
|
||||||
def get_id(self):
|
def get_id(self):
|
||||||
return str(self._nm_device.get_op())
|
return str(self._nm_device.get_op())
|
||||||
|
|
||||||
|
def get_nm_device(self):
|
||||||
|
return self._nm_device
|
||||||
|
|
||||||
|
@ -15,18 +15,28 @@
|
|||||||
# along with this program; if not, write to the Free Software
|
# along with this program; if not, write to the Free Software
|
||||||
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
# Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA
|
||||||
|
|
||||||
|
from gettext import gettext as _
|
||||||
|
|
||||||
|
import gtk
|
||||||
|
|
||||||
from sugar import profile
|
from sugar import profile
|
||||||
from sugar.graphics.icon import CanvasIcon
|
from sugar.graphics.icon import CanvasIcon
|
||||||
from sugar.graphics import style
|
from sugar.graphics import style
|
||||||
from model.devices import device
|
from model.devices import device
|
||||||
|
|
||||||
|
from sugar.graphics.palette import Palette
|
||||||
|
from model.devices.network import wireless
|
||||||
|
|
||||||
class DeviceView(CanvasIcon):
|
class DeviceView(CanvasIcon):
|
||||||
def __init__(self, model):
|
def __init__(self, model):
|
||||||
CanvasIcon.__init__(self, size=style.MEDIUM_ICON_SIZE,
|
CanvasIcon.__init__(self, size=style.MEDIUM_ICON_SIZE,
|
||||||
icon_name='network-mesh')
|
icon_name='network-mesh')
|
||||||
self._model = model
|
self._model = model
|
||||||
|
self._palette = MeshPalette(_("Mesh Network"), model)
|
||||||
|
self.set_palette(self._palette)
|
||||||
|
|
||||||
model.connect('notify::state', self._state_changed_cb)
|
model.connect('notify::state', self._state_changed_cb)
|
||||||
|
model.connect('notify::activation-stage', self._state_changed_cb)
|
||||||
self._update_state()
|
self._update_state()
|
||||||
|
|
||||||
def _state_changed_cb(self, model, pspec):
|
def _state_changed_cb(self, model, pspec):
|
||||||
@ -35,6 +45,8 @@ class DeviceView(CanvasIcon):
|
|||||||
def _update_state(self):
|
def _update_state(self):
|
||||||
# FIXME Change icon colors once we have real icons
|
# FIXME Change icon colors once we have real icons
|
||||||
state = self._model.props.state
|
state = self._model.props.state
|
||||||
|
self._palette.update_state(state)
|
||||||
|
|
||||||
if state == device.STATE_ACTIVATING:
|
if state == device.STATE_ACTIVATING:
|
||||||
self.props.fill_color = style.COLOR_INACTIVE_FILL.get_svg()
|
self.props.fill_color = style.COLOR_INACTIVE_FILL.get_svg()
|
||||||
self.props.stroke_color = style.COLOR_INACTIVE_STROKE.get_svg()
|
self.props.stroke_color = style.COLOR_INACTIVE_STROKE.get_svg()
|
||||||
@ -43,3 +55,69 @@ class DeviceView(CanvasIcon):
|
|||||||
elif state == device.STATE_INACTIVE:
|
elif state == device.STATE_INACTIVE:
|
||||||
self.props.fill_color = style.COLOR_INACTIVE_FILL.get_svg()
|
self.props.fill_color = style.COLOR_INACTIVE_FILL.get_svg()
|
||||||
self.props.stroke_color = style.COLOR_INACTIVE_STROKE.get_svg()
|
self.props.stroke_color = style.COLOR_INACTIVE_STROKE.get_svg()
|
||||||
|
|
||||||
|
if state == device.STATE_INACTIVE:
|
||||||
|
self._palette.set_primary_text(_("Mesh Network"))
|
||||||
|
else:
|
||||||
|
chan = wireless.freq_to_channel(self._model.props.frequency)
|
||||||
|
if chan > 0:
|
||||||
|
self._palette.set_primary_text(_("Mesh Network") + " %d" % chan)
|
||||||
|
self._palette.set_mesh_step(self._model.props.mesh_step, state)
|
||||||
|
|
||||||
|
class MeshPalette(Palette):
|
||||||
|
def __init__(self, primary_text, model):
|
||||||
|
Palette.__init__(self, primary_text, menu_after_content=True)
|
||||||
|
self._model = model
|
||||||
|
|
||||||
|
self._step_label = gtk.Label()
|
||||||
|
self._step_label.show()
|
||||||
|
|
||||||
|
vbox = gtk.VBox()
|
||||||
|
vbox.pack_start(self._step_label)
|
||||||
|
vbox.show()
|
||||||
|
|
||||||
|
self.set_content(vbox)
|
||||||
|
|
||||||
|
self._disconnect_item = gtk.MenuItem(_('Disconnect...'))
|
||||||
|
self._disconnect_item.connect('activate', self._disconnect_activate_cb)
|
||||||
|
self.menu.append(self._disconnect_item)
|
||||||
|
|
||||||
|
def update_state(self, state):
|
||||||
|
if state == device.STATE_ACTIVATED:
|
||||||
|
self._disconnect_item.show()
|
||||||
|
else:
|
||||||
|
self._disconnect_item.hide()
|
||||||
|
|
||||||
|
def _disconnect_activate_cb(self, menuitem):
|
||||||
|
# Disconnection for an mesh means activating the default mesh device
|
||||||
|
# again without a channel
|
||||||
|
network_manager = hardwaremanager.get_network_manager()
|
||||||
|
nm_device = self._model.get_nm_device()
|
||||||
|
if network_manager and nm_device:
|
||||||
|
network_manager.set_active_device(nm_device)
|
||||||
|
|
||||||
|
def set_mesh_step(self, step, state):
|
||||||
|
label = ""
|
||||||
|
if step == 1:
|
||||||
|
if state == device.STATE_ACTIVATED:
|
||||||
|
label = _("Connected to a School Mesh Portal")
|
||||||
|
elif state == device.STATE_ACTIVATING:
|
||||||
|
label = _("Looking for a School Mesh Portal...")
|
||||||
|
elif step == 3:
|
||||||
|
if state == device.STATE_ACTIVATED:
|
||||||
|
label = _("Connected to an XO Mesh Portal")
|
||||||
|
elif state == device.STATE_ACTIVATING:
|
||||||
|
label = _("Looking for an XO Mesh Portal...")
|
||||||
|
elif step == 4:
|
||||||
|
if state == device.STATE_ACTIVATED:
|
||||||
|
label = _("Connected to a Simple Mesh")
|
||||||
|
elif state == device.STATE_ACTIVATING:
|
||||||
|
label = _("Starting a Simple Mesh")
|
||||||
|
|
||||||
|
if len(label):
|
||||||
|
self._step_label.set_text(label)
|
||||||
|
else:
|
||||||
|
import logging
|
||||||
|
logging.debug("Unhandled mesh step %d" % step)
|
||||||
|
self._step_label.set_text(_("Unknown Mesh"))
|
||||||
|
|
||||||
|
@ -27,13 +27,24 @@ from sugar.graphics.palette import Palette
|
|||||||
from model.devices.network import wireless
|
from model.devices.network import wireless
|
||||||
from model.devices import device
|
from model.devices import device
|
||||||
|
|
||||||
|
from hardware import hardwaremanager
|
||||||
|
from hardware import nmclient
|
||||||
|
|
||||||
_ICON_NAME = 'network-wireless'
|
_ICON_NAME = 'network-wireless'
|
||||||
|
|
||||||
class DeviceView(CanvasIcon):
|
class DeviceView(CanvasIcon):
|
||||||
def __init__(self, model):
|
def __init__(self, model):
|
||||||
CanvasIcon.__init__(self, size=style.MEDIUM_ICON_SIZE)
|
CanvasIcon.__init__(self, size=style.MEDIUM_ICON_SIZE)
|
||||||
self._model = model
|
self._model = model
|
||||||
self._palette = WirelessPalette(self._get_palette_primary_text())
|
|
||||||
|
meshdev = None
|
||||||
|
network_manager = hardwaremanager.get_network_manager()
|
||||||
|
for device in network_manager.get_devices():
|
||||||
|
if device.get_type() == nmclient.DEVICE_TYPE_802_11_MESH_OLPC:
|
||||||
|
meshdev = device
|
||||||
|
break
|
||||||
|
|
||||||
|
self._palette = WirelessPalette(self._get_palette_primary_text(), meshdev)
|
||||||
self.set_palette(self._palette)
|
self.set_palette(self._palette)
|
||||||
self._counter = 0
|
self._counter = 0
|
||||||
self._palette.set_frequency(self._model.props.frequency)
|
self._palette.set_frequency(self._model.props.frequency)
|
||||||
@ -87,8 +98,9 @@ class DeviceView(CanvasIcon):
|
|||||||
self.props.stroke_color = style.COLOR_INACTIVE_STROKE.get_svg()
|
self.props.stroke_color = style.COLOR_INACTIVE_STROKE.get_svg()
|
||||||
|
|
||||||
class WirelessPalette(Palette):
|
class WirelessPalette(Palette):
|
||||||
def __init__(self, primary_text):
|
def __init__(self, primary_text, meshdev):
|
||||||
Palette.__init__(self, primary_text)
|
Palette.__init__(self, primary_text, menu_after_content=True)
|
||||||
|
self._meshdev = meshdev
|
||||||
|
|
||||||
self._chan_label = gtk.Label()
|
self._chan_label = gtk.Label()
|
||||||
self._chan_label.show()
|
self._chan_label.show()
|
||||||
@ -97,16 +109,23 @@ class WirelessPalette(Palette):
|
|||||||
vbox.pack_start(self._chan_label)
|
vbox.pack_start(self._chan_label)
|
||||||
vbox.show()
|
vbox.show()
|
||||||
|
|
||||||
|
if meshdev:
|
||||||
|
disconnect_item = gtk.MenuItem(_('Disconnect...'))
|
||||||
|
disconnect_item.connect('activate', self._disconnect_activate_cb)
|
||||||
|
self.menu.append(disconnect_item)
|
||||||
|
disconnect_item.show()
|
||||||
|
|
||||||
self.set_content(vbox)
|
self.set_content(vbox)
|
||||||
|
|
||||||
|
def _disconnect_activate_cb(self, menuitem):
|
||||||
|
# Disconnection for an AP means activating the default mesh device
|
||||||
|
network_manager = hardwaremanager.get_network_manager()
|
||||||
|
if network_manager and self._meshdev:
|
||||||
|
network_manager.set_active_device(self._meshdev)
|
||||||
|
|
||||||
def set_frequency(self, freq):
|
def set_frequency(self, freq):
|
||||||
chans = { 2.412: 1, 2.417: 2, 2.422: 3, 2.427: 4,
|
|
||||||
2.432: 5, 2.437: 6, 2.442: 7, 2.447: 8,
|
|
||||||
2.452: 9, 2.457: 10, 2.462: 11, 2.467: 12,
|
|
||||||
2.472: 13
|
|
||||||
}
|
|
||||||
try:
|
try:
|
||||||
chan = chans[freq]
|
chan = wireless.freq_to_channel(freq)
|
||||||
except KeyError:
|
except KeyError:
|
||||||
chan = 0
|
chan = 0
|
||||||
self._chan_label.set_text("%s: %d" % (_("Channel"), chan))
|
self._chan_label.set_text("%s: %d" % (_("Channel"), chan))
|
||||||
|
@ -18,6 +18,7 @@ import random
|
|||||||
|
|
||||||
import hippo
|
import hippo
|
||||||
import gobject
|
import gobject
|
||||||
|
import gtk
|
||||||
from gettext import gettext as _
|
from gettext import gettext as _
|
||||||
|
|
||||||
from sugar.graphics.spreadlayout import SpreadLayout
|
from sugar.graphics.spreadlayout import SpreadLayout
|
||||||
@ -26,10 +27,12 @@ from sugar.graphics import style
|
|||||||
from sugar.graphics import xocolor
|
from sugar.graphics import xocolor
|
||||||
from sugar.graphics.icon import get_icon_state
|
from sugar.graphics.icon import get_icon_state
|
||||||
from sugar.graphics import style
|
from sugar.graphics import style
|
||||||
|
from sugar.graphics import palette
|
||||||
from sugar import profile
|
from sugar import profile
|
||||||
|
|
||||||
from model import accesspointmodel
|
from model import accesspointmodel
|
||||||
from model.devices.network import mesh
|
from model.devices.network import mesh
|
||||||
|
from model.devices.network import wireless
|
||||||
from hardware import hardwaremanager
|
from hardware import hardwaremanager
|
||||||
from hardware import nmclient
|
from hardware import nmclient
|
||||||
from view.BuddyIcon import BuddyIcon
|
from view.BuddyIcon import BuddyIcon
|
||||||
@ -42,9 +45,11 @@ from hardware.nmclient import NM_802_11_CAP_PROTO_WEP, NM_802_11_CAP_PROTO_WPA,
|
|||||||
_ICON_NAME = 'network-wireless'
|
_ICON_NAME = 'network-wireless'
|
||||||
|
|
||||||
class AccessPointView(PulsingIcon):
|
class AccessPointView(PulsingIcon):
|
||||||
def __init__(self, model):
|
def __init__(self, model, mesh_device=None):
|
||||||
PulsingIcon.__init__(self, size=style.MEDIUM_ICON_SIZE, cache=True)
|
PulsingIcon.__init__(self, size=style.MEDIUM_ICON_SIZE, cache=True)
|
||||||
self._model = model
|
self._model = model
|
||||||
|
self._meshdev = mesh_device
|
||||||
|
self._disconnect_item = None
|
||||||
|
|
||||||
self.connect('activated', self._activate_cb)
|
self.connect('activated', self._activate_cb)
|
||||||
|
|
||||||
@ -56,6 +61,9 @@ class AccessPointView(PulsingIcon):
|
|||||||
self._device_stroke = stroke
|
self._device_stroke = stroke
|
||||||
self._device_fill = fill
|
self._device_fill = fill
|
||||||
|
|
||||||
|
self._palette = self._create_palette()
|
||||||
|
self.set_palette(self._palette)
|
||||||
|
|
||||||
self._update_icon()
|
self._update_icon()
|
||||||
self._update_name()
|
self._update_name()
|
||||||
self._update_state()
|
self._update_state()
|
||||||
@ -67,6 +75,28 @@ class AccessPointView(PulsingIcon):
|
|||||||
elif (caps & NM_802_11_CAP_PROTO_WEP) or (caps & NM_802_11_CAP_PROTO_WPA) or (caps & NM_802_11_CAP_PROTO_WPA2):
|
elif (caps & NM_802_11_CAP_PROTO_WEP) or (caps & NM_802_11_CAP_PROTO_WPA) or (caps & NM_802_11_CAP_PROTO_WPA2):
|
||||||
self.props.badge_name = "badge-locked"
|
self.props.badge_name = "badge-locked"
|
||||||
|
|
||||||
|
def _create_palette(self):
|
||||||
|
p = palette.Palette(self._model.props.name, menu_after_content=True)
|
||||||
|
if not self._meshdev:
|
||||||
|
return p
|
||||||
|
|
||||||
|
# Only show disconnect when there's a mesh device, because mesh takes
|
||||||
|
# priority over the normal wireless device. NM doesn't have a "disconnect"
|
||||||
|
# method for a device either (for various reasons) so this doesn't
|
||||||
|
# have a good mapping
|
||||||
|
self._disconnect_item = gtk.MenuItem(_('Disconnect...'))
|
||||||
|
self._disconnect_item.connect('activate', self._disconnect_activate_cb)
|
||||||
|
p.menu.append(self._disconnect_item)
|
||||||
|
if self._model.props.state == accesspointmodel.STATE_CONNECTED:
|
||||||
|
self._disconnect_item.show()
|
||||||
|
return p
|
||||||
|
|
||||||
|
def _disconnect_activate_cb(self, menuitem):
|
||||||
|
# Disconnection for an AP means activating the default mesh device
|
||||||
|
network_manager = hardwaremanager.get_network_manager()
|
||||||
|
if network_manager and self._meshdev:
|
||||||
|
network_manager.set_active_device(self._meshdev)
|
||||||
|
|
||||||
def _strength_changed_cb(self, model, pspec):
|
def _strength_changed_cb(self, model, pspec):
|
||||||
self._update_icon()
|
self._update_icon()
|
||||||
|
|
||||||
@ -84,10 +114,7 @@ class AccessPointView(PulsingIcon):
|
|||||||
network_manager.set_active_device(device, network)
|
network_manager.set_active_device(device, network)
|
||||||
|
|
||||||
def _update_name(self):
|
def _update_name(self):
|
||||||
if self.palette:
|
self._palette.set_primary_text(self._model.props.name)
|
||||||
self.palette.set_primary_text(self._model.props.name)
|
|
||||||
else:
|
|
||||||
self.set_tooltip(self._model.props.name)
|
|
||||||
|
|
||||||
def _update_icon(self):
|
def _update_icon(self):
|
||||||
icon_name = get_icon_state(_ICON_NAME, self._model.props.strength)
|
icon_name = get_icon_state(_ICON_NAME, self._model.props.strength)
|
||||||
@ -96,6 +123,8 @@ class AccessPointView(PulsingIcon):
|
|||||||
|
|
||||||
def _update_state(self):
|
def _update_state(self):
|
||||||
if self._model.props.state == accesspointmodel.STATE_CONNECTING:
|
if self._model.props.state == accesspointmodel.STATE_CONNECTING:
|
||||||
|
if self._disconnect_item:
|
||||||
|
self._disconnect_item.hide()
|
||||||
self.props.pulse_time = 1.0
|
self.props.pulse_time = 1.0
|
||||||
self.props.colors = [
|
self.props.colors = [
|
||||||
[ style.Color(self._device_stroke).get_svg(),
|
[ style.Color(self._device_stroke).get_svg(),
|
||||||
@ -104,6 +133,8 @@ class AccessPointView(PulsingIcon):
|
|||||||
'#e2e2e2' ]
|
'#e2e2e2' ]
|
||||||
]
|
]
|
||||||
elif self._model.props.state == accesspointmodel.STATE_CONNECTED:
|
elif self._model.props.state == accesspointmodel.STATE_CONNECTED:
|
||||||
|
if self._disconnect_item:
|
||||||
|
self._disconnect_item.show()
|
||||||
self.props.pulse_time = 2.0
|
self.props.pulse_time = 2.0
|
||||||
self.props.colors = [
|
self.props.colors = [
|
||||||
[ style.Color(self._device_stroke).get_svg(),
|
[ style.Color(self._device_stroke).get_svg(),
|
||||||
@ -112,6 +143,8 @@ class AccessPointView(PulsingIcon):
|
|||||||
style.Color(self._device_fill).get_svg() ]
|
style.Color(self._device_fill).get_svg() ]
|
||||||
]
|
]
|
||||||
elif self._model.props.state == accesspointmodel.STATE_NOTCONNECTED:
|
elif self._model.props.state == accesspointmodel.STATE_NOTCONNECTED:
|
||||||
|
if self._disconnect_item:
|
||||||
|
self._disconnect_item.hide()
|
||||||
self.props.pulse_time = 0.0
|
self.props.pulse_time = 0.0
|
||||||
self.props.colors = [
|
self.props.colors = [
|
||||||
[ style.Color(self._device_stroke).get_svg(),
|
[ style.Color(self._device_stroke).get_svg(),
|
||||||
@ -122,11 +155,20 @@ class AccessPointView(PulsingIcon):
|
|||||||
_MESH_ICON_NAME = 'network-mesh'
|
_MESH_ICON_NAME = 'network-mesh'
|
||||||
|
|
||||||
class MeshDeviceView(PulsingIcon):
|
class MeshDeviceView(PulsingIcon):
|
||||||
def __init__(self, nm_device):
|
def __init__(self, nm_device, channel):
|
||||||
|
if not channel in [1, 6, 11]:
|
||||||
|
raise ValueError("Invalid channel %d" % channel)
|
||||||
|
|
||||||
PulsingIcon.__init__(self, size=style.MEDIUM_ICON_SIZE,
|
PulsingIcon.__init__(self, size=style.MEDIUM_ICON_SIZE,
|
||||||
icon_name=_MESH_ICON_NAME, cache=True)
|
icon_name=_MESH_ICON_NAME, cache=True)
|
||||||
|
|
||||||
self._nm_device = nm_device
|
self._nm_device = nm_device
|
||||||
self.set_tooltip(_("Mesh Network"))
|
self.channel = channel
|
||||||
|
self.props.badge_name = "badge-channel-%d" % self.channel
|
||||||
|
|
||||||
|
self._disconnect_item = None
|
||||||
|
self._palette = self._create_palette()
|
||||||
|
self.set_palette(self._palette)
|
||||||
|
|
||||||
mycolor = profile.get_color()
|
mycolor = profile.get_color()
|
||||||
self._device_fill = mycolor.get_fill_color()
|
self._device_fill = mycolor.get_fill_color()
|
||||||
@ -135,39 +177,66 @@ class MeshDeviceView(PulsingIcon):
|
|||||||
self.connect('activated', self._activate_cb)
|
self.connect('activated', self._activate_cb)
|
||||||
|
|
||||||
self._nm_device.connect('state-changed', self._state_changed_cb)
|
self._nm_device.connect('state-changed', self._state_changed_cb)
|
||||||
|
self._nm_device.connect('activation-stage-changed', self._state_changed_cb)
|
||||||
self._update_state()
|
self._update_state()
|
||||||
|
|
||||||
|
def _create_palette(self):
|
||||||
|
p = palette.Palette(_("Mesh Network") + " " + str(self.channel), menu_after_content=True)
|
||||||
|
|
||||||
|
self._disconnect_item = gtk.MenuItem(_('Disconnect...'))
|
||||||
|
self._disconnect_item.connect('activate', self._disconnect_activate_cb)
|
||||||
|
p.menu.append(self._disconnect_item)
|
||||||
|
|
||||||
|
state = self._nm_device.get_state()
|
||||||
|
chan = wireless.freq_to_channel(self._nm_device.get_frequency())
|
||||||
|
if state == nmclient.DEVICE_STATE_ACTIVATED and chan == self.channel:
|
||||||
|
self._disconnect_item.show()
|
||||||
|
return p
|
||||||
|
|
||||||
|
def _disconnect_activate_cb(self, menuitem):
|
||||||
|
network_manager = hardwaremanager.get_network_manager()
|
||||||
|
if network_manager:
|
||||||
|
network_manager.set_active_device(self._nm_device)
|
||||||
|
|
||||||
def _activate_cb(self, icon):
|
def _activate_cb(self, icon):
|
||||||
network_manager = hardwaremanager.get_network_manager()
|
network_manager = hardwaremanager.get_network_manager()
|
||||||
network_manager.set_active_device(self._nm_device)
|
if network_manager:
|
||||||
|
freq = wireless.channel_to_freq(self.channel)
|
||||||
|
network_manager.set_active_device(self._nm_device, mesh_freq=freq)
|
||||||
|
|
||||||
def _state_changed_cb(self, model):
|
def _state_changed_cb(self, model):
|
||||||
self._update_state()
|
self._update_state()
|
||||||
|
|
||||||
def _update_state(self):
|
def _update_state(self):
|
||||||
state = self._nm_device.get_state()
|
state = self._nm_device.get_state()
|
||||||
if state == nmclient.DEVICE_STATE_ACTIVATING:
|
chan = wireless.freq_to_channel(self._nm_device.get_frequency())
|
||||||
|
if state == nmclient.DEVICE_STATE_ACTIVATING and chan == self.channel:
|
||||||
|
self._disconnect_item.hide()
|
||||||
self.props.pulse_time = 0.75
|
self.props.pulse_time = 0.75
|
||||||
self.props.colors = [
|
self.props.colors = [
|
||||||
[ style.Color(self._device_stroke).get_svg(),
|
[ style.Color(self._device_stroke).get_svg(),
|
||||||
style.Color(self._device_fill).get_svg() ],
|
style.Color(self._device_fill).get_svg() ],
|
||||||
[ style.Color(self._device_stroke).get_svg(),
|
[ style.Color(self._device_stroke).get_svg(),
|
||||||
'#e2e2e2' ]
|
'#e2e2e2' ]
|
||||||
]
|
]
|
||||||
elif state == nmclient.DEVICE_STATE_ACTIVATED:
|
elif state == nmclient.DEVICE_STATE_ACTIVATED and chan == self.channel:
|
||||||
|
self._disconnect_item.show()
|
||||||
self.props.pulse_time = 1.5
|
self.props.pulse_time = 1.5
|
||||||
self.props.colors = [
|
self.props.colors = [
|
||||||
[ style.Color(self._device_stroke).get_svg(),
|
[ style.Color(self._device_stroke).get_svg(),
|
||||||
style.Color(self._device_fill).get_svg() ],
|
style.Color(self._device_fill).get_svg() ],
|
||||||
[ '#ffffff',
|
[ '#ffffff',
|
||||||
style.Color(self._device_fill).get_svg() ]
|
style.Color(self._device_fill).get_svg() ]
|
||||||
]
|
]
|
||||||
elif state == nmclient.DEVICE_STATE_INACTIVE:
|
elif state == nmclient.DEVICE_STATE_INACTIVE or chan != self.channel:
|
||||||
|
self._disconnect_item.hide()
|
||||||
self.props.pulse_time = 0.0
|
self.props.pulse_time = 0.0
|
||||||
self.props.colors = [
|
self.props.colors = [
|
||||||
[ style.Color(self._device_stroke).get_svg(),
|
[ style.Color(self._device_stroke).get_svg(),
|
||||||
style.Color(self._device_fill).get_svg() ]
|
style.Color(self._device_fill).get_svg() ]
|
||||||
]
|
]
|
||||||
|
else:
|
||||||
|
raise RuntimeError("Shouldn't get here")
|
||||||
|
|
||||||
class ActivityView(hippo.CanvasBox):
|
class ActivityView(hippo.CanvasBox):
|
||||||
def __init__(self, shell, model):
|
def __init__(self, shell, model):
|
||||||
@ -216,7 +285,7 @@ class MeshBox(hippo.CanvasBox):
|
|||||||
self._buddies = {}
|
self._buddies = {}
|
||||||
self._activities = {}
|
self._activities = {}
|
||||||
self._access_points = {}
|
self._access_points = {}
|
||||||
self._mesh = None
|
self._mesh = {}
|
||||||
self._buddy_to_activity = {}
|
self._buddy_to_activity = {}
|
||||||
self._suspended = True
|
self._suspended = True
|
||||||
|
|
||||||
@ -245,18 +314,22 @@ class MeshBox(hippo.CanvasBox):
|
|||||||
self._access_point_removed_cb)
|
self._access_point_removed_cb)
|
||||||
|
|
||||||
if self._model.get_mesh():
|
if self._model.get_mesh():
|
||||||
self._add_mesh_icon(self._model.get_mesh())
|
self._mesh_added_cb(self._model, self._model.get_mesh())
|
||||||
|
|
||||||
self._model.connect('mesh-added',
|
self._model.connect('mesh-added',
|
||||||
self._mesh_added_cb)
|
self._mesh_added_cb)
|
||||||
self._model.connect('mesh-removed',
|
self._model.connect('mesh-removed',
|
||||||
self._mesh_removed_cb)
|
self._mesh_removed_cb)
|
||||||
|
|
||||||
def _mesh_added_cb(self, model, mesh):
|
def _mesh_added_cb(self, model, meshdev):
|
||||||
self._add_mesh_icon(mesh)
|
self._add_mesh_icon(meshdev, 1)
|
||||||
|
self._add_mesh_icon(meshdev, 6)
|
||||||
|
self._add_mesh_icon(meshdev, 11)
|
||||||
|
|
||||||
def _mesh_removed_cb(self, model):
|
def _mesh_removed_cb(self, model):
|
||||||
self._remove_mesh()
|
self._remove_mesh_icon(1)
|
||||||
|
self._remove_mesh_icon(6)
|
||||||
|
self._remove_mesh_icon(11)
|
||||||
|
|
||||||
def _buddy_added_cb(self, model, buddy_model):
|
def _buddy_added_cb(self, model, buddy_model):
|
||||||
self._add_alone_buddy(buddy_model)
|
self._add_alone_buddy(buddy_model)
|
||||||
@ -282,19 +355,19 @@ class MeshBox(hippo.CanvasBox):
|
|||||||
def _access_point_removed_cb(self, model, ap_model):
|
def _access_point_removed_cb(self, model, ap_model):
|
||||||
self._remove_access_point(ap_model)
|
self._remove_access_point(ap_model)
|
||||||
|
|
||||||
def _add_mesh_icon(self, mesh):
|
def _add_mesh_icon(self, meshdev, channel):
|
||||||
if self._mesh:
|
if self._mesh.has_key(channel):
|
||||||
self._remove_mesh()
|
self._remove_mesh_icon(channel)
|
||||||
if not mesh:
|
if not meshdev:
|
||||||
return
|
return
|
||||||
self._mesh = MeshDeviceView(mesh)
|
self._mesh[channel] = MeshDeviceView(meshdev, channel)
|
||||||
self._layout.add(self._mesh)
|
self._layout.add(self._mesh[channel])
|
||||||
|
|
||||||
def _remove_mesh(self):
|
def _remove_mesh_icon(self, channel):
|
||||||
if not self._mesh:
|
if not self._mesh.has_key(channel):
|
||||||
return
|
return
|
||||||
self._layout.remove(self._mesh)
|
self._layout.remove(self._mesh[channel])
|
||||||
self._mesh = None
|
del self._mesh[channel]
|
||||||
|
|
||||||
def _add_alone_buddy(self, buddy_model):
|
def _add_alone_buddy(self, buddy_model):
|
||||||
icon = BuddyIcon(self._shell, buddy_model)
|
icon = BuddyIcon(self._shell, buddy_model)
|
||||||
@ -344,7 +417,8 @@ class MeshBox(hippo.CanvasBox):
|
|||||||
del self._activities[activity_model.get_id()]
|
del self._activities[activity_model.get_id()]
|
||||||
|
|
||||||
def _add_access_point(self, ap_model):
|
def _add_access_point(self, ap_model):
|
||||||
icon = AccessPointView(ap_model)
|
meshdev = self._model.get_mesh()
|
||||||
|
icon = AccessPointView(ap_model, meshdev)
|
||||||
self._layout.add(icon)
|
self._layout.add(icon)
|
||||||
|
|
||||||
self._access_points[ap_model.get_id()] = icon
|
self._access_points[ap_model.get_id()] = icon
|
||||||
|
Loading…
Reference in New Issue
Block a user