Source code for armarx.remote_gui.client

"""
This module provides functionality for remote GUI clients.

Classes:
 - Tab: Base class for remote tabs.
 - Client: Remote GUI client.
"""

from armarx import RemoteGuiInterfacePrx
from armarx import RemoteGui as rg

from armarx.remote_gui.ice_wrapper import TabProxy
from armarx.remote_gui.widgets import Widget

from typing import Callable, List, Optional
import abc


[docs]class Tab: """ This is a base class for remote GUI tabs. A remote GUI tab is identified by an id string. The structure of the GUI is defined via a single root widget. The root widget is usually a container widget which contains many child widgets. The :meth:`create_widget_tree` method must be overridden in derived class to create the root widget and all its children. The :meth:`on_update` method is called by the Client after updates have been received from the GUI. """ def __init__(self, id: str): self.id = id self.root_widget = None self.proxy: TabProxy = None
[docs] @abc.abstractmethod def create_widget_tree(self) -> Widget: """ Creates a widget tree and returns the root widget of the tree. This method MUST be overridden by a derived class. In a derived class, you will create your widgets here and nest them via container widgets. You can store the widgets, which you want to access later (usually value widgets), as member variables. Widgets which are only used for layout or to display constant values can be created as local variables and need not be stored. """ raise NotImplementedError()
[docs] @abc.abstractmethod def on_update(self): """ Updates the internal state after an update has been received from the GUI. This method CAN be overridden by a derived class. In a derived class, you can update your internal state according to the new updates from the GUI. """ pass
[docs] def connect(self, proxy: "RemoteGuiInterfacePrx"): """Connect this instance with a remote GUI tab.""" self.root_widget = self.create_widget_tree() proxy.createTab(self.id, self.root_widget.desc) self.proxy = TabProxy(proxy, self.id) self.proxy.initialize_widget(self.root_widget)
[docs] def remove(self): """Remove this remote GUI tab.""" self.proxy.remove()
[docs] def receive_updates(self): """Receive updates from the remote GUI.""" self.proxy.receive_updates()
[docs] def send_updates(self): """Send updates to the remote GUI.""" self.proxy.send_updates()
[docs]class Client: """The client connects to the remote GUI and manages tabs.""" def __init__(self, provider_name: str = "RemoteGuiProvider"): self.provider_name = provider_name self.proxy = RemoteGuiInterfacePrx.get_proxy(provider_name) self.tabs: List[Tab] = []
[docs] def add_tab(self, tab: Tab): """Add a tab and establish a connection to the remote GUI.""" tab.connect(self.proxy) self.tabs.append(tab)
[docs] def remove_tab(self, tab: Tab): """Remove a tab from the remote GUI.""" tab.remove() self.tabs.remove(tab)
[docs] def receive_updates(self): """Receive updates from the remote GUI for all added tabs.""" for tab in self.tabs: tab.receive_updates() tab.on_update()
[docs] def send_updates(self): """Send updates to the remote GUI for all added tabs.""" for tab in self.tabs: tab.send_updates()
[docs] def update_loop( self, callback: Callable, block=True ) -> Optional["threading.Thread"]: """ Run a loop receiving and sending updates. :param callback: The callback to call after receiving updates. :param block: If true, blocks until a KeyboardInterrupt is received. If false, starts the loop in a thread and returns the thread. :return: If block is False, returns the thread, otherwise returns nothing. """ if block: try: while True: try: self.receive_updates() callback() self.send_updates() except KeyError: pass except KeyboardInterrupt: pass else: from threading import Thread thread = Thread( target=lambda: self.update_loop(callback=callback, block=True), name="Remote GUI", ) thread.start() return thread