# # # from __future__ import absolute_import, division, print_function, unicode_literals from gi.repository import Gtk, GLib from logging import getLogger, DEBUG as _DEBUG _log = getLogger(__name__) del getLogger from . import icons as _icons from logitech.unifying_receiver import status as _status # # # _PAIRING_TIMEOUT = 30 def _create_page(assistant, kind, header=None, icon_name=None, text=None): p = Gtk.VBox(False, 8) assistant.append_page(p) assistant.set_page_type(p, kind) if header: item = Gtk.HBox(False, 16) p.pack_start(item, False, True, 0) label = Gtk.Label(header) label.set_alignment(0, 0) label.set_line_wrap(True) item.pack_start(label, True, True, 0) if icon_name: icon = Gtk.Image.new_from_icon_name(icon_name, Gtk.IconSize.DIALOG) icon.set_alignment(1, 0) item.pack_start(icon, False, False, 0) if text: label = Gtk.Label(text) label.set_alignment(0, 0) label.set_line_wrap(True) p.pack_start(label, False, False, 0) p.show_all() return p # def _fake_device(receiver): # from logitech.unifying_receiver import PairedDevice # dev = PairedDevice(receiver, 6) # dev._wpid = '1234' # dev._kind = 'touchpad' # dev._codename = 'T650' # dev._name = 'Wireless Rechargeable Touchpad T650' # dev._serial = '0123456789' # dev._protocol = 2.0 # dev.status = _status.DeviceStatus(dev, lambda *foo: None) # dev.status['encrypted'] = False # return dev def _check_lock_state(assistant, receiver): if not assistant.is_drawable(): if _log.isEnabledFor(_DEBUG): _log.debug("assistant %s destroyed, bailing out", assistant) return False if receiver.status.get(_status.ERROR): # receiver.status.new_device = _fake_device(receiver) _pairing_failed(assistant, receiver, receiver.status.pop(_status.ERROR)) return False if receiver.status.new_device: device, receiver.status.new_device = receiver.status.new_device, None _pairing_succeeded(assistant, receiver, device) return False if not receiver.status.lock_open: _pairing_failed(assistant, receiver, 'failed to open pairing lock') return False return True def _prepare(assistant, page, receiver): index = assistant.get_current_page() if _log.isEnabledFor(_DEBUG): _log.debug("prepare %s %d %s", assistant, index, page) if index == 0: if receiver.set_lock(False, timeout=_PAIRING_TIMEOUT): assert receiver.status.new_device is None assert receiver.status.get(_status.ERROR) is None spinner = page.get_children()[-1] spinner.start() GLib.timeout_add(750, _check_lock_state, assistant, receiver) assistant.set_page_complete(page, True) else: GLib.idle_add(_pairing_failed, assistant, receiver, 'the pairing lock did not open') else: assistant.remove_page(0) def _finish(assistant, receiver): if _log.isEnabledFor(_DEBUG): _log.debug("finish %s", assistant) assistant.destroy() receiver.status.new_device = None if receiver.status.lock_open: receiver.set_lock() else: receiver.status[_status.ERROR] = None def _pairing_failed(assistant, receiver, error): if _log.isEnabledFor(_DEBUG): _log.debug("%s fail: %s", receiver, error) assistant.commit() header = 'Pairing failed: %s.' % error if 'timeout' in str(error): text = 'Make sure your device is within range,\nand it has a decent battery charge.' else: text = None _create_page(assistant, Gtk.AssistantPageType.SUMMARY, header, 'dialog-error', text) assistant.next_page() assistant.commit() def _pairing_succeeded(assistant, receiver, device): assert device if _log.isEnabledFor(_DEBUG): _log.debug("%s success: %s", receiver, device) page = _create_page(assistant, Gtk.AssistantPageType.SUMMARY) header = Gtk.Label('Found a new device:') header.set_alignment(0.5, 0) page.pack_start(header, False, False, 0) device_icon = Gtk.Image() icon_set = _icons.device_icon_set(device.name, device.kind) device_icon.set_from_icon_set(icon_set, Gtk.IconSize.LARGE) device_icon.set_alignment(0.5, 1) page.pack_start(device_icon, True, True, 0) device_label = Gtk.Label() device_label.set_markup('%s' % device.name) device_label.set_alignment(0.5, 0) page.pack_start(device_label, True, True, 0) hbox = Gtk.HBox(False, 8) hbox.pack_start(Gtk.Label(' '), False, False, 0) hbox.set_property('expand', False) hbox.set_property('halign', Gtk.Align.CENTER) page.pack_start(hbox, False, False, 0) def _check_encrypted(dev): if assistant.is_drawable(): if device.status.get('encrypted') == False: hbox.pack_start(Gtk.Image.new_from_icon_name('security-low', Gtk.IconSize.MENU), False, False, 0) hbox.pack_start(Gtk.Label('The wireless link is not encrypted!'), False, False, 0) hbox.show_all() else: return True GLib.timeout_add(500, _check_encrypted, device) page.show_all() assistant.next_page() assistant.commit() def create(receiver): assert receiver is not None assert receiver.kind is None assistant = Gtk.Assistant() assistant.set_title(receiver.name + ': pair new device') assistant.set_icon_name('list-add') assistant.set_size_request(400, 240) assistant.set_resizable(False) assistant.set_role('pair-device') page_intro = _create_page(assistant, Gtk.AssistantPageType.PROGRESS, 'Turn on the device you want to pair.', 'preferences-desktop-peripherals', 'If the device is already turned on,\nturn if off and on again.') spinner = Gtk.Spinner() spinner.set_visible(True) page_intro.pack_end(spinner, True, True, 24) assistant.connect('prepare', _prepare, receiver) assistant.connect('cancel', _finish, receiver) assistant.connect('close', _finish, receiver) return assistant