redesigned the window UI
This commit is contained in:
		
							parent
							
								
									0255194b46
								
							
						
					
					
						commit
						5347f41d73
					
				|  | @ -61,6 +61,8 @@ if __name__ == '__main__': | ||||||
| 	if args.systray: | 	if args.systray: | ||||||
| 		tray_icon = ui.icon.create(APP_TITLE, (ui.window.toggle, window)) | 		tray_icon = ui.icon.create(APP_TITLE, (ui.window.toggle, window)) | ||||||
| 		tray_icon.set_from_icon_name(APP_TITLE + '-fail') | 		tray_icon.set_from_icon_name(APP_TITLE + '-fail') | ||||||
|  | 	else: | ||||||
|  | 		window.present() | ||||||
| 
 | 
 | ||||||
| 	Gtk.main() | 	Gtk.main() | ||||||
| 
 | 
 | ||||||
|  |  | ||||||
							
								
								
									
										218
									
								
								app/ui/window.py
								
								
								
								
							
							
						
						
									
										218
									
								
								app/ui/window.py
								
								
								
								
							|  | @ -7,8 +7,8 @@ from gi.repository import (Gtk, Gdk) | ||||||
| from logitech.devices import constants as C | from logitech.devices import constants as C | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| _DEVICE_ICON_SIZE = Gtk.IconSize.DND | _DEVICE_ICON_SIZE = Gtk.IconSize.DIALOG | ||||||
| _STATUS_ICON_SIZE = Gtk.IconSize.DIALOG | _STATUS_ICON_SIZE = Gtk.IconSize.DND | ||||||
| _PLACEHOLDER = '~' | _PLACEHOLDER = '~' | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | @ -36,10 +36,10 @@ def _find_children(container, *child_names): | ||||||
| 	return result if count > 1 else result[0] | 	return result if count > 1 else result[0] | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def _update_receiver_box(box, receiver): | def _update_receiver_box(box, rstatus): | ||||||
| 	label, buttons_box = _find_children(box, 'receiver-status', 'receiver-buttons') | 	label, buttons_box = _find_children(box, 'label', 'buttons') | ||||||
| 	label.set_text(receiver.text or '') | 	label.set_text(rstatus.text or '') | ||||||
| 	buttons_box.set_visible(receiver.code >= C.STATUS.CONNECTED) | 	buttons_box.set_visible(rstatus.code >= C.STATUS.CONNECTED) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def _update_device_box(frame, devstatus): | def _update_device_box(frame, devstatus): | ||||||
|  | @ -48,147 +48,133 @@ def _update_device_box(frame, devstatus): | ||||||
| 		frame.set_name(_PLACEHOLDER) | 		frame.set_name(_PLACEHOLDER) | ||||||
| 		return | 		return | ||||||
| 
 | 
 | ||||||
|  | 	icon, label = _find_children(frame, 'icon', 'label') | ||||||
|  | 
 | ||||||
| 	frame.set_visible(True) | 	frame.set_visible(True) | ||||||
| 	if frame.get_name() != devstatus.name: | 	if frame.get_name() != devstatus.name: | ||||||
| 		frame.set_name(devstatus.name) | 		frame.set_name(devstatus.name) | ||||||
| 		icon = _find_children(frame, 'device-icon') |  | ||||||
| 		icon.set_from_icon_name(devstatus.name, _DEVICE_ICON_SIZE) | 		icon.set_from_icon_name(devstatus.name, _DEVICE_ICON_SIZE) | ||||||
| 		icon.set_tooltip_text(devstatus.name) | 		icon.set_tooltip_text(devstatus.name) | ||||||
|  | 		label.set_markup('<b>' + devstatus.name + '</b>') | ||||||
| 
 | 
 | ||||||
| 	expander = _find_children(frame, 'device-expander') | 	status = _find_children(frame, 'status') | ||||||
| 	if devstatus.code < C.STATUS.CONNECTED: | 	if devstatus.code < C.STATUS.CONNECTED: | ||||||
| 		expander.set_sensitive(False) | 		icon.set_sensitive(False) | ||||||
| 		expander.set_expanded(False) | 		label.set_sensitive(False) | ||||||
| 		expander.set_label('<big><b>%s</b></big>\n%s' % (devstatus.name, devstatus.text)) | 		status.set_visible(False) | ||||||
| 		return | 		return | ||||||
| 
 | 
 | ||||||
| 	expander.set_sensitive(True) | 	icon.set_sensitive(True) | ||||||
| 	status_icons = expander.get_child().get_children() | 	label.set_sensitive(True) | ||||||
|  | 	status.set_visible(True) | ||||||
|  | 	status_icons = status.get_children() | ||||||
| 
 | 
 | ||||||
| 	texts = [] | 	battery_icon, battery_label = status_icons[0:2] | ||||||
| 
 |  | ||||||
| 	light_icon = status_icons[-2] |  | ||||||
| 	light_level = getattr(devstatus, C.PROPS.LIGHT_LEVEL, None) |  | ||||||
| 	if light_level is None: |  | ||||||
| 		light_icon.set_visible(False) |  | ||||||
| 	else: |  | ||||||
| 		light_icon.set_visible(True) |  | ||||||
| 		icon_name = 'light_%02d' % (20 * ((light_level + 50) // 100)) |  | ||||||
| 		light_icon.set_from_icon_name(icon_name, _STATUS_ICON_SIZE) |  | ||||||
| 		tooltip = 'Light: %d lux' % light_level |  | ||||||
| 		light_icon.set_tooltip_text(tooltip) |  | ||||||
| 		texts.append(tooltip) |  | ||||||
| 
 |  | ||||||
| 	battery_icon = status_icons[-1] |  | ||||||
| 	battery_level = getattr(devstatus, C.PROPS.BATTERY_LEVEL, None) | 	battery_level = getattr(devstatus, C.PROPS.BATTERY_LEVEL, None) | ||||||
| 	if battery_level is None: | 	if battery_level is None: | ||||||
| 		battery_icon.set_sensitive(False) | 		battery_icon.set_sensitive(False) | ||||||
| 		battery_icon.set_from_icon_name('battery_unknown', _STATUS_ICON_SIZE) | 		battery_icon.set_from_icon_name('battery_unknown', _STATUS_ICON_SIZE) | ||||||
| 		battery_icon.set_tooltip_text('Battery: unknown') | 		battery_label.set_sensitive(False) | ||||||
|  | 		battery_label.set_text('') | ||||||
| 	else: | 	else: | ||||||
| 		battery_icon.set_sensitive(True) | 		battery_icon.set_sensitive(True) | ||||||
| 		icon_name = 'battery_%02d' % (20 * ((battery_level + 10) // 20)) | 		icon_name = 'battery_%02d' % (20 * ((battery_level + 10) // 20)) | ||||||
| 		battery_icon.set_from_icon_name(icon_name, _STATUS_ICON_SIZE) | 		battery_icon.set_from_icon_name(icon_name, _STATUS_ICON_SIZE) | ||||||
| 		tooltip = 'Battery: %d%%' % battery_level | 		battery_label.set_sensitive(True) | ||||||
| 		battery_icon.set_tooltip_text(tooltip) | 		battery_label.set_text('%d%%' % battery_level) | ||||||
| 		texts.append(tooltip) |  | ||||||
| 
 | 
 | ||||||
| 	battery_status = getattr(devstatus, C.PROPS.BATTERY_STATUS, None) | 	battery_status = getattr(devstatus, C.PROPS.BATTERY_STATUS, None) | ||||||
| 	if battery_status is not None: | 	if battery_status is None: | ||||||
| 		texts.append(battery_status) | 		battery_icon.set_tooltip_text('') | ||||||
| 		battery_icon.set_tooltip_text(battery_icon.get_tooltip_text() + '\n' + battery_status) |  | ||||||
| 
 |  | ||||||
| 	if texts: |  | ||||||
| 		expander.set_label('<big><b>%s</b></big>\n%s' % (devstatus.name, ', '.join(texts))) |  | ||||||
| 	else: | 	else: | ||||||
| 		expander.set_label('<big><b>%s</b></big>\n%s' % (devstatus.name, devstatus.text)) | 		battery_icon.set_tooltip_text(battery_status) | ||||||
|  | 
 | ||||||
|  | 	light_icon, light_label = status_icons[2:4] | ||||||
|  | 	light_level = getattr(devstatus, C.PROPS.LIGHT_LEVEL, None) | ||||||
|  | 	if light_level is None: | ||||||
|  | 		light_icon.set_visible(False) | ||||||
|  | 		light_label.set_visible(False) | ||||||
|  | 	else: | ||||||
|  | 		light_icon.set_visible(True) | ||||||
|  | 		icon_name = 'light_%02d' % (20 * ((light_level + 50) // 100)) | ||||||
|  | 		light_icon.set_from_icon_name(icon_name, _STATUS_ICON_SIZE) | ||||||
|  | 		light_label.set_visible(True) | ||||||
|  | 		light_label.set_text('%d lux' % light_level) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def update(window, receiver, devices, icon_name=None): | def update(window, rstatus, devices, icon_name=None): | ||||||
| 	if window and window.get_child(): | 	if window and window.get_child(): | ||||||
| 		if icon_name is not None: | 		if icon_name is not None: | ||||||
| 			window.set_icon_name(icon_name) | 			window.set_icon_name(icon_name) | ||||||
| 
 | 
 | ||||||
| 		vbox = window.get_child().get_child() | 		vbox = window.get_child().get_child() | ||||||
| 		controls = list(vbox.get_children()) | 		controls = list(vbox.get_children()) | ||||||
| 		_update_receiver_box(controls[0], receiver) | 		_update_receiver_box(controls[0], rstatus) | ||||||
| 		for index in range(1, len(controls)): | 		for index in range(1, len(controls)): | ||||||
| 			_update_device_box(controls[index], devices.get(index)) | 			_update_device_box(controls[index], devices.get(index)) | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def _receiver_box(rstatus): | def _device_box(name=None, has_status_icons=True, has_frame=True): | ||||||
| 	box = Gtk.HBox(homogeneous=False, spacing=8) | 	box = Gtk.HBox(homogeneous=False, spacing=10) | ||||||
| 	box.set_border_width(4) |  | ||||||
| 
 |  | ||||||
| 	icon = Gtk.Image.new_from_icon_name(rstatus.name, _DEVICE_ICON_SIZE) |  | ||||||
| 	icon.set_alignment(0.5, 0) |  | ||||||
| 	icon.set_tooltip_text(rstatus.name) |  | ||||||
| 	box.pack_start(icon, False, False, 0) |  | ||||||
| 
 |  | ||||||
| 	label = Gtk.Label('Initializing...') |  | ||||||
| 	label.set_alignment(0, 0.5) |  | ||||||
| 	label.set_name('receiver-status') |  | ||||||
| 	box.pack_start(label, True, True, 0) |  | ||||||
| 
 |  | ||||||
| 	toolbar = Gtk.Toolbar() |  | ||||||
| 	toolbar.set_style(Gtk.ToolbarStyle.ICONS) |  | ||||||
| 	toolbar.set_name('receiver-buttons') |  | ||||||
| 	toolbar.set_show_arrow(False) |  | ||||||
| 	toolbar.set_icon_size(Gtk.IconSize.BUTTON) |  | ||||||
| 	box.pack_end(toolbar, False, False, 0) |  | ||||||
| 
 |  | ||||||
| 	def _action(button, function, params): |  | ||||||
| 		button.set_sensitive(False) |  | ||||||
| 		function(button, *params) |  | ||||||
| 		button.set_sensitive(True) |  | ||||||
| 
 |  | ||||||
| 	def _add_button(name, icon, action): |  | ||||||
| 		button = Gtk.ToolButton() |  | ||||||
| 		button.set_icon_name(icon) |  | ||||||
| 		button.set_tooltip_text(name) |  | ||||||
| 		if action: |  | ||||||
| 			function = action[0] |  | ||||||
| 			params = action[1:] |  | ||||||
| 			button.connect('clicked', _action, function, params) |  | ||||||
| 		else: |  | ||||||
| 			button.set_sensitive(False) |  | ||||||
| 		toolbar.insert(button, -1) |  | ||||||
| 
 |  | ||||||
| 	_add_button('Scan for devices', 'reload', rstatus.refresh) |  | ||||||
| 	_add_button('Pair new device', 'add', rstatus.pair) |  | ||||||
| 
 |  | ||||||
| 	box.show_all() |  | ||||||
| 	toolbar.set_visible(False) |  | ||||||
| 	return box |  | ||||||
| 
 |  | ||||||
| 
 |  | ||||||
| def _device_box(): |  | ||||||
| 	box = Gtk.HBox(homogeneous=False, spacing=8) |  | ||||||
| 	box.set_border_width(4) | 	box.set_border_width(4) | ||||||
| 
 | 
 | ||||||
| 	icon = Gtk.Image() | 	icon = Gtk.Image() | ||||||
|  | 	if name: | ||||||
|  | 		icon.set_from_icon_name(name, _DEVICE_ICON_SIZE) | ||||||
|  | 		icon.set_tooltip_text(name) | ||||||
|  | 	else: | ||||||
|  | 		icon.set_from_icon_name('dialog-question', _DEVICE_ICON_SIZE) | ||||||
| 	icon.set_alignment(0.5, 0) | 	icon.set_alignment(0.5, 0) | ||||||
| 	icon.set_name('device-icon') | 	icon.set_name('icon') | ||||||
| 	box.pack_start(icon, False, False, 0) | 	box.pack_start(icon, False, False, 0) | ||||||
| 
 | 
 | ||||||
| 	expander = Gtk.Expander() | 	vbox = Gtk.VBox(homogeneous=False, spacing=8) | ||||||
| 	expander.set_use_markup(True) | 	box.pack_start(vbox, True, True, 0) | ||||||
| 	expander.set_spacing(4) | 
 | ||||||
| 	expander.set_name('device-expander') | 	label = Gtk.Label('...') | ||||||
| 	box.pack_start(expander, True, True, 1) | 	label.set_alignment(0, 0.5) | ||||||
|  | 	label.set_name('label') | ||||||
|  | 
 | ||||||
|  | 	status_box = Gtk.HBox(homogeneous=False, spacing=0) | ||||||
|  | 	status_box.set_name('status') | ||||||
|  | 
 | ||||||
|  | 	if has_status_icons: | ||||||
|  | 		vbox.pack_start(label, True, True, 0) | ||||||
| 
 | 
 | ||||||
| 	ebox = Gtk.HBox(False, 8) |  | ||||||
| 		battery_icon = Gtk.Image.new_from_icon_name('battery_unknown', _STATUS_ICON_SIZE) | 		battery_icon = Gtk.Image.new_from_icon_name('battery_unknown', _STATUS_ICON_SIZE) | ||||||
| 	ebox.pack_end(battery_icon, False, True, 0) | 		status_box.pack_start(battery_icon, False, True, 0) | ||||||
| 	light_icon = Gtk.Image.new_from_icon_name('light_unknown', _STATUS_ICON_SIZE) | 		battery_label = Gtk.Label() | ||||||
| 	ebox.pack_end(light_icon, False, True, 0) | 		battery_label.set_width_chars(6) | ||||||
| 	expander.add(ebox) | 		battery_label.set_alignment(0, 0.5) | ||||||
|  | 		status_box.pack_start(battery_label, False, True, 0) | ||||||
| 
 | 
 | ||||||
|  | 		light_icon = Gtk.Image.new_from_icon_name('light_unknown', _STATUS_ICON_SIZE) | ||||||
|  | 		status_box.pack_start(light_icon, False, True, 0) | ||||||
|  | 		light_label = Gtk.Label() | ||||||
|  | 		light_label.set_alignment(0, 0.5) | ||||||
|  | 		light_label.set_width_chars(8) | ||||||
|  | 		status_box.pack_start(light_label, False, True, 0) | ||||||
|  | 	else: | ||||||
|  | 		status_box.pack_start(label, True, True, 0) | ||||||
|  | 
 | ||||||
|  | 		toolbar = Gtk.Toolbar() | ||||||
|  | 		toolbar.set_style(Gtk.ToolbarStyle.ICONS) | ||||||
|  | 		toolbar.set_icon_size(Gtk.IconSize.MENU) | ||||||
|  | 		toolbar.set_name('buttons') | ||||||
|  | 		toolbar.set_show_arrow(False) | ||||||
|  | 		status_box.pack_end(toolbar, False, False, 0) | ||||||
|  | 
 | ||||||
|  | 	vbox.pack_start(status_box, True, True, 0) | ||||||
|  | 
 | ||||||
|  | 	box.show_all() | ||||||
|  | 
 | ||||||
|  | 	if has_frame: | ||||||
| 		frame = Gtk.Frame() | 		frame = Gtk.Frame() | ||||||
| 		frame.add(box) | 		frame.add(box) | ||||||
| 	frame.show_all() |  | ||||||
| 	frame.set_visible(False) |  | ||||||
| 		return frame | 		return frame | ||||||
|  | 	else: | ||||||
|  | 		toolbar.set_visible(False) | ||||||
|  | 		return box | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def create(title, rstatus, systray=False): | def create(title, rstatus, systray=False): | ||||||
|  | @ -199,9 +185,11 @@ def create(title, rstatus, systray=False): | ||||||
| 	vbox = Gtk.VBox(homogeneous=False, spacing=4) | 	vbox = Gtk.VBox(homogeneous=False, spacing=4) | ||||||
| 	vbox.set_border_width(4) | 	vbox.set_border_width(4) | ||||||
| 
 | 
 | ||||||
| 	vbox.add(_receiver_box(rstatus)) | 	rbox = _device_box(rstatus.name, False, False) | ||||||
|  | 	vbox.add(rbox) | ||||||
| 	for i in range(1, 1 + rstatus.max_devices): | 	for i in range(1, 1 + rstatus.max_devices): | ||||||
| 		vbox.add(_device_box()) | 		dbox = _device_box() | ||||||
|  | 		vbox.add(dbox) | ||||||
| 	vbox.set_visible(True) | 	vbox.set_visible(True) | ||||||
| 
 | 
 | ||||||
| 	frame = Gtk.Frame() | 	frame = Gtk.Frame() | ||||||
|  | @ -211,7 +199,7 @@ def create(title, rstatus, systray=False): | ||||||
| 	window.add(frame) | 	window.add(frame) | ||||||
| 
 | 
 | ||||||
| 	geometry = Gdk.Geometry() | 	geometry = Gdk.Geometry() | ||||||
| 	geometry.min_width = 300 | 	geometry.min_width = 260 | ||||||
| 	geometry.min_height = 40 | 	geometry.min_height = 40 | ||||||
| 	window.set_geometry_hints(frame, geometry, Gdk.WindowHints.MIN_SIZE) | 	window.set_geometry_hints(frame, geometry, Gdk.WindowHints.MIN_SIZE) | ||||||
| 
 | 
 | ||||||
|  | @ -219,23 +207,35 @@ def create(title, rstatus, systray=False): | ||||||
| 	window.set_default_size(geometry.min_width, geometry.min_height) | 	window.set_default_size(geometry.min_width, geometry.min_height) | ||||||
| 
 | 
 | ||||||
| 	if systray: | 	if systray: | ||||||
|  | 		def _state_event(window, event): | ||||||
|  | 			if event.new_window_state & Gdk.WindowState.ICONIFIED: | ||||||
|  | 				# position = window.get_position() | ||||||
|  | 				window.hide() | ||||||
|  | 				window.deiconify() | ||||||
|  | 				# window.move(*position) | ||||||
|  | 				return True | ||||||
|  | 
 | ||||||
| 		window.set_keep_above(True) | 		window.set_keep_above(True) | ||||||
| 		window.set_deletable(False) | 		# window.set_deletable(False) | ||||||
| 		window.set_decorated(False) | 		# window.set_decorated(False) | ||||||
| 		window.set_position(Gtk.WindowPosition.MOUSE) | 		window.set_position(Gtk.WindowPosition.MOUSE) | ||||||
| 		window.set_type_hint(Gdk.WindowTypeHint.MENU) | 		# window.set_type_hint(Gdk.WindowTypeHint.MENU) | ||||||
| 		window.set_skip_taskbar_hint(True) | 		window.set_skip_taskbar_hint(True) | ||||||
| 		window.set_skip_pager_hint(True) | 		window.set_skip_pager_hint(True) | ||||||
|  | 
 | ||||||
|  | 		window.connect('window-state-event', _state_event) | ||||||
|  | 		window.connect('delete-event', lambda w, e: toggle(None, window) or True) | ||||||
| 	else: | 	else: | ||||||
| 		window.set_position(Gtk.WindowPosition.CENTER) | 		window.set_position(Gtk.WindowPosition.CENTER) | ||||||
| 		window.connect('delete-event', Gtk.main_quit) | 		window.connect('delete-event', Gtk.main_quit) | ||||||
| 		window.present() |  | ||||||
| 
 | 
 | ||||||
| 	return window | 	return window | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
| def toggle(_, window): | def toggle(_, window): | ||||||
| 	if window.get_visible(): | 	if window.get_visible(): | ||||||
|  | 		# position = window.get_position() | ||||||
| 		window.hide() | 		window.hide() | ||||||
|  | 		# window.move(*position) | ||||||
| 	else: | 	else: | ||||||
| 		window.present() | 		window.present() | ||||||
|  |  | ||||||
|  | @ -43,7 +43,7 @@ class Watcher(Thread): | ||||||
| 	def __init__(self, status_changed_callback, notify_callback=None): | 	def __init__(self, status_changed_callback, notify_callback=None): | ||||||
| 		super(Watcher, self).__init__(group='Solaar', name='Watcher') | 		super(Watcher, self).__init__(group='Solaar', name='Watcher') | ||||||
| 		self.daemon = True | 		self.daemon = True | ||||||
| 		self.active = False | 		self._active = False | ||||||
| 
 | 
 | ||||||
| 		self.notify = notify_callback | 		self.notify = notify_callback | ||||||
| 		self.status_text = None | 		self.status_text = None | ||||||
|  | @ -59,9 +59,9 @@ class Watcher(Thread): | ||||||
| 		self.devices = {} | 		self.devices = {} | ||||||
| 
 | 
 | ||||||
| 	def run(self): | 	def run(self): | ||||||
| 		self.active = True | 		self._active = True | ||||||
| 
 | 
 | ||||||
| 		while self.active: | 		while self._active: | ||||||
| 			if self.listener is None: | 			if self.listener is None: | ||||||
| 				self._update_status_text() | 				self._update_status_text() | ||||||
| 
 | 
 | ||||||
|  | @ -97,7 +97,7 @@ class Watcher(Thread): | ||||||
| 				self._update_status_text() | 				self._update_status_text() | ||||||
| 
 | 
 | ||||||
| 			for i in range(0, int(_THREAD_SLEEP / _SLEEP_QUANT)): | 			for i in range(0, int(_THREAD_SLEEP / _SLEEP_QUANT)): | ||||||
| 				if self.active: | 				if self._active: | ||||||
| 					time.sleep(_SLEEP_QUANT) | 					time.sleep(_SLEEP_QUANT) | ||||||
| 				else: | 				else: | ||||||
| 					break | 					break | ||||||
|  | @ -107,7 +107,8 @@ class Watcher(Thread): | ||||||
| 			self.listener = None | 			self.listener = None | ||||||
| 
 | 
 | ||||||
| 	def stop(self): | 	def stop(self): | ||||||
| 		self.active = False | 		if self._active: | ||||||
|  | 			self._active = False | ||||||
| 			self.join() | 			self.join() | ||||||
| 
 | 
 | ||||||
| 	def _request_status(self, devstatus): | 	def _request_status(self, devstatus): | ||||||
|  | @ -127,7 +128,7 @@ class Watcher(Thread): | ||||||
| 		return updated | 		return updated | ||||||
| 
 | 
 | ||||||
| 	def _new_device(self, dev): | 	def _new_device(self, dev): | ||||||
| 		if not self.active: | 		if not self._active: | ||||||
| 			return None | 			return None | ||||||
| 
 | 
 | ||||||
| 		if type(dev) == int: | 		if type(dev) == int: | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue