124 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			Python
		
	
	
	
			
		
		
	
	
			124 lines
		
	
	
		
			2.6 KiB
		
	
	
	
		
			Python
		
	
	
	
| #
 | |
| #
 | |
| #
 | |
| 
 | |
| from __future__ import absolute_import, division, print_function, unicode_literals
 | |
| 
 | |
| 
 | |
| from logging import getLogger, DEBUG as _DEBUG
 | |
| _log = getLogger(__name__)
 | |
| del getLogger
 | |
| 
 | |
| from gi.repository import GLib, Gtk
 | |
| GLib.threads_init()
 | |
| 
 | |
| 
 | |
| def _error_dialog(reason, object):
 | |
| 	_log.error("%s: %s", reason, object)
 | |
| 
 | |
| 	if reason == 'permissions':
 | |
| 		title = 'Permissions error'
 | |
| 		text = ('Found a Logitech Receiver (%s), but did not have permission to open it.\n'
 | |
| 				'\n'
 | |
| 				'If you\'ve just installed Solaar, try removing the receiver\n'
 | |
| 				'and plugging it back in.' % object)
 | |
| 	elif reason == 'unpair':
 | |
| 		title = 'Unpairing failed'
 | |
| 		text = ('Failed to unpair device\n%s .' % object)
 | |
| 	else:
 | |
| 		raise Exception("ui.error_dialog: don't know how to handle (%s, %s)", reason, object)
 | |
| 
 | |
| 	assert title
 | |
| 	assert text
 | |
| 
 | |
| 	m = Gtk.MessageDialog(None, Gtk.DialogFlags.MODAL, Gtk.MessageType.ERROR, Gtk.ButtonsType.CLOSE, text)
 | |
| 	m.set_title(title)
 | |
| 	m.run()
 | |
| 	m.destroy()
 | |
| 
 | |
| 
 | |
| def error_dialog(reason, object):
 | |
| 	assert reason is not None
 | |
| 	GLib.idle_add(_error_dialog, reason, object)
 | |
| 
 | |
| #
 | |
| # 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
 | |
| except ImportError:
 | |
| 	from queue import Queue
 | |
| _task_queue = Queue(16)
 | |
| del Queue
 | |
| 
 | |
| 
 | |
| from threading import Thread, current_thread as _current_thread
 | |
| 
 | |
| def _process_async_queue():
 | |
| 	t = _current_thread()
 | |
| 	t.alive = True
 | |
| 	while t.alive:
 | |
| 		function, args, kwargs = _task_queue.get()
 | |
| 		if function:
 | |
| 			function(*args, **kwargs)
 | |
| 	if _log.isEnabledFor(_DEBUG):
 | |
| 		_log.debug("stopped %s", t.name)
 | |
| 
 | |
| _queue_processor = Thread(name='AsyncUI', target=_process_async_queue)
 | |
| _queue_processor.daemon = True
 | |
| _queue_processor.alive = False
 | |
| _queue_processor.start()
 | |
| 
 | |
| del Thread
 | |
| 
 | |
| def async(function, *args, **kwargs):
 | |
| 	task = (function, args, kwargs)
 | |
| 	_task_queue.put(task)
 | |
| 
 | |
| 
 | |
| #
 | |
| #
 | |
| #
 | |
| 
 | |
| from . import notify, tray, window
 | |
| 
 | |
| def init():
 | |
| 	notify.init()
 | |
| 	tray.init()
 | |
| 	window.init()
 | |
| 
 | |
| def run_loop():
 | |
| 	Gtk.main()
 | |
| 
 | |
| def destroy():
 | |
| 	# stop the async UI processor
 | |
| 	_queue_processor.alive = False
 | |
| 	async(None)
 | |
| 
 | |
| 	tray.destroy()
 | |
| 	window.destroy()
 | |
| 	notify.uninit()
 | |
| 
 | |
| 
 | |
| from logitech.unifying_receiver.status import ALERT
 | |
| def _status_changed(device, alert, reason):
 | |
| 	assert device is not None
 | |
| 	if _log.isEnabledFor(_DEBUG):
 | |
| 		_log.debug("status changed: %s (%s) %s", device, alert, reason)
 | |
| 
 | |
| 	tray.update(device)
 | |
| 	if alert & ALERT.ATTENTION:
 | |
| 		tray.attention(reason)
 | |
| 
 | |
| 	need_popup = alert & (ALERT.SHOW_WINDOW | ALERT.ATTENTION)
 | |
| 	window.update(device, need_popup)
 | |
| 
 | |
| 	if alert & ALERT.NOTIFICATION:
 | |
| 		notify.show(device, reason)
 | |
| 
 | |
| 
 | |
| def status_changed(device, alert=ALERT.NONE, reason=None):
 | |
| 	GLib.idle_add(_status_changed, device, alert, reason)
 |