205 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			205 lines
		
	
	
		
			4.9 KiB
		
	
	
	
		
			Python
		
	
	
	
| #
 | |
| #
 | |
| #
 | |
| 
 | |
| from __future__ import absolute_import, division, print_function, unicode_literals
 | |
| 
 | |
| from gi.repository import Gtk, GLib
 | |
| 
 | |
| from logitech.unifying_receiver.settings import KIND as _SETTING_KIND
 | |
| 
 | |
| #
 | |
| # a separate thread is used to read/write from the device
 | |
| # so as not to block the main (GUI) thread
 | |
| #
 | |
| 
 | |
| try:
 | |
| 	from Queue import Queue as _Queue
 | |
| except ImportError:
 | |
| 	from queue import Queue as _Queue
 | |
| _apply_queue = _Queue(8)
 | |
| 
 | |
| def _process_apply_queue():
 | |
| 	def _write_start(sbox):
 | |
| 		_, failed, spinner, control = sbox.get_children()
 | |
| 		control.set_sensitive(False)
 | |
| 		failed.set_visible(False)
 | |
| 		spinner.set_visible(True)
 | |
| 		spinner.start()
 | |
| 
 | |
| 	while True:
 | |
| 		task = _apply_queue.get()
 | |
| 		assert isinstance(task, tuple)
 | |
| 		device_is_online = True
 | |
| 		# print ("task", *task)
 | |
| 		if task[0] == 'write':
 | |
| 			_, setting, value, sbox = task
 | |
| 			GLib.idle_add(_write_start, sbox, priority=0)
 | |
| 			value = setting.write(value)
 | |
| 		elif task[0] == 'read':
 | |
| 			_, setting, force_read, sbox, device_is_online = task
 | |
| 			value = setting.read(not force_read)
 | |
| 		GLib.idle_add(_update_setting_item, sbox, value, device_is_online, priority=99)
 | |
| 
 | |
| from threading import Thread as _Thread
 | |
| _queue_processor = _Thread(name='SettingsProcessor', target=_process_apply_queue)
 | |
| _queue_processor.daemon = True
 | |
| _queue_processor.start()
 | |
| 
 | |
| #
 | |
| #
 | |
| #
 | |
| 
 | |
| def _create_toggle_control(setting):
 | |
| 	def _switch_notify(switch, _, s):
 | |
| 		if switch.get_sensitive():
 | |
| 			_apply_queue.put(('write', s, switch.get_active() == True, switch.get_parent()))
 | |
| 
 | |
| 	c = Gtk.Switch()
 | |
| 	c.connect('notify::active', _switch_notify, setting)
 | |
| 	return c
 | |
| 
 | |
| def _create_choice_control(setting):
 | |
| 	def _combo_notify(cbbox, s):
 | |
| 		if cbbox.get_sensitive():
 | |
| 			_apply_queue.put(('write', s, cbbox.get_active_id(), cbbox.get_parent()))
 | |
| 
 | |
| 	c = Gtk.ComboBoxText()
 | |
| 	for entry in setting.choices:
 | |
| 		c.append(str(entry), str(entry))
 | |
| 	c.connect('changed', _combo_notify, setting)
 | |
| 	return c
 | |
| 
 | |
| # def _create_slider_control(setting):
 | |
| # 	def _slider_notify(slider, s):
 | |
| # 		if slider.get_sensitive():
 | |
| # 			_apply_queue.put(('write', s, slider.get_value(), slider.get_parent()))
 | |
| #
 | |
| # 	c = Gtk.Scale(setting.choices)
 | |
| # 	c.connect('value-changed', _slider_notify, setting)
 | |
| #
 | |
| # 	return c
 | |
| 
 | |
| #
 | |
| #
 | |
| #
 | |
| 
 | |
| def _create_sbox(s):
 | |
| 	sbox = Gtk.HBox(homogeneous=False, spacing=6)
 | |
| 	sbox.pack_start(Gtk.Label(s.label), False, False, 0)
 | |
| 
 | |
| 	spinner = Gtk.Spinner()
 | |
| 	spinner.set_tooltip_text('Working...')
 | |
| 
 | |
| 	failed = Gtk.Image.new_from_icon_name('dialog-warning', Gtk.IconSize.SMALL_TOOLBAR)
 | |
| 	failed.set_tooltip_text('Failed to read value from the device.')
 | |
| 
 | |
| 	if s.kind == _SETTING_KIND.toggle:
 | |
| 		control = _create_toggle_control(s)
 | |
| 	elif s.kind == _SETTING_KIND.choice:
 | |
| 		control = _create_choice_control(s)
 | |
| 	# elif s.kind == _SETTING_KIND.range:
 | |
| 	# 	control = _create_slider_control(s)
 | |
| 	else:
 | |
| 		raise NotImplemented
 | |
| 
 | |
| 	control.set_sensitive(False)  # the first read will enable it
 | |
| 	sbox.pack_end(control, False, False, 0)
 | |
| 	sbox.pack_end(spinner, False, False, 0)
 | |
| 	sbox.pack_end(failed, False, False, 0)
 | |
| 
 | |
| 	if s.description:
 | |
| 		sbox.set_tooltip_text(s.description)
 | |
| 
 | |
| 	sbox.show_all()
 | |
| 	spinner.start()  # the first read will stop it
 | |
| 	failed.set_visible(False)
 | |
| 
 | |
| 	return sbox
 | |
| 
 | |
| 
 | |
| def _update_setting_item(sbox, value, is_online=True):
 | |
| 	_, failed, spinner, control = sbox.get_children()
 | |
| 	spinner.set_visible(False)
 | |
| 	spinner.stop()
 | |
| 
 | |
| 	# print ("update", control, "with new value", value)
 | |
| 	if value is None:
 | |
| 		control.set_sensitive(False)
 | |
| 		failed.set_visible(is_online)
 | |
| 		return
 | |
| 
 | |
| 	failed.set_visible(False)
 | |
| 	if isinstance(control, Gtk.Switch):
 | |
| 		control.set_active(value)
 | |
| 	elif isinstance(control, Gtk.ComboBoxText):
 | |
| 		control.set_active_id(str(value))
 | |
| 	# elif isinstance(control, Gtk.Scale):
 | |
| 	# 	control.set_value(int(value))
 | |
| 	else:
 | |
| 		raise NotImplemented
 | |
| 	control.set_sensitive(True)
 | |
| 
 | |
| #
 | |
| #
 | |
| #
 | |
| 
 | |
| # config panel
 | |
| _box = None
 | |
| _items = {}
 | |
| 
 | |
| def create():
 | |
| 	global _box
 | |
| 	assert _box is None
 | |
| 	_box = Gtk.VBox(homogeneous=False, spacing=8)
 | |
| 	_box._last_device = None
 | |
| 	return _box
 | |
| 
 | |
| 
 | |
| def update(device, is_online=None):
 | |
| 	assert _box is not None
 | |
| 	assert device
 | |
| 	device_id = (device.receiver.path, device.number)
 | |
| 	if is_online is None:
 | |
| 		is_online = bool(device.online)
 | |
| 
 | |
| 	# if the device changed since last update, clear the box first
 | |
| 	if device_id != _box._last_device:
 | |
| 		_box.set_visible(False)
 | |
| 		_box._last_device = device_id
 | |
| 
 | |
| 	# hide controls belonging to other devices
 | |
| 	for k, sbox in _items.items():
 | |
| 		sbox = _items[k]
 | |
| 		sbox.set_visible(k[0:2] == device_id)
 | |
| 
 | |
| 	for s in device.settings:
 | |
| 		k = (device_id[0], device_id[1], s.name)
 | |
| 		if k in _items:
 | |
| 			sbox = _items[k]
 | |
| 		else:
 | |
| 			sbox = _items[k] = _create_sbox(s)
 | |
| 			_box.pack_start(sbox, False, False, 0)
 | |
| 
 | |
| 		_apply_queue.put(('read', s, False, sbox, is_online))
 | |
| 
 | |
| 	_box.set_visible(True)
 | |
| 
 | |
| 
 | |
| def clean(device):
 | |
| 	"""Remove the controls for a given device serial.
 | |
| 	Needed after the device has been unpaired.
 | |
| 	"""
 | |
| 	assert _box is not None
 | |
| 	device_id = (device.receiver.path, device.number)
 | |
| 	for k in list(_items.keys()):
 | |
| 		if k[0:2] == device_id:
 | |
| 			_box.remove(_items[k])
 | |
| 			del _items[k]
 | |
| 
 | |
| 
 | |
| def destroy():
 | |
| 	global _box
 | |
| 	_box = None
 | |
| 	_items.clear()
 |