Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: Added QWidget #3

Open
wants to merge 2 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion plugin.xml
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
<library path="src">
<class name="RqtEmbedWindow" type="rqt_embed_window.RqtEmbedWindow.RqtEmbedWindow"
<class name="EmbedWindow" type="rqt_embed_window.embed_window.EmbedWindow"
base_class_type="rqt_gui_py::Plugin">
<description>
A GUI plugin to execute GUI applications as rqt_ plugins
Expand Down
4 changes: 2 additions & 2 deletions scripts/rqt_embed_window
Original file line number Diff line number Diff line change
Expand Up @@ -2,8 +2,8 @@

import sys

from rqt_embed_window.RqtEmbedWindow import RqtEmbedWindow
from rqt_embed_window.embed_window import EmbedWindow
from rqt_gui.main import Main

main = Main(filename='rqt_embed_window')
sys.exit(main.main(standalone='rqt_embed_window'))
sys.exit(main.main(standalone='rqt_embed_window.embed_window.EmbedWindow'))
1 change: 0 additions & 1 deletion scripts/test_if_window_can_be_embedded.py
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,6 @@ def run_app(window_id):

main_widget.show()
app.exec_()



if __name__ == '__main__':
Expand Down
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
<?xml version="1.0" encoding="UTF-8"?>
<ui version="4.0">
<class>RqtEmbedWindowWidget</class>
<widget class="QWidget" name="RqtEmbedWindowWidget">
<class>EmbedWindowWidget</class>
<widget class="QWidget" name="EmbedWindowWidget">
<property name="enabled">
<bool>true</bool>
</property>
Expand Down
Empty file modified src/rqt_embed_window/__init__.py
100644 → 100755
Empty file.
77 changes: 77 additions & 0 deletions src/rqt_embed_window/embed_window.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,77 @@
from python_qt_binding.QtWidgets import QInputDialog
from qt_gui.plugin import Plugin
from qt_gui.settings import Settings

from .embed_window_widget import EmbedWindowWidget


class EmbedWindow(Plugin):
"""
This plugin allows to embed a Qt window into a rqt plugin.
"""

def __init__(self, context):
super(EmbedWindow, self).__init__(context)
self.setObjectName('EmbedWindow')
self._command = ''
self._window_name = ''
self._process = None
self._timeout_to_window_discovery = 20.0

# Create widget
self._widget = EmbedWindowWidget()
# Add widget to the user interface
context.add_widget(self._widget)

self.context = context

def add_external_window_widget(self):
self._widget.add_external_window_widget(self._command, self._window_name, self._timeout_to_window_discovery)

# Give the title (for rqt_gui compositions) some information
if self.context.serial_number() < 2:
self._widget.setWindowTitle('{} ({})'.format(self._widget.windowTitle(), self._command))
else:
self._widget.setWindowTitle('{} ({}) ({})'.format(self._widget.windowTitle(),
self.context.serial_number(), self._command))

def shutdown_plugin(self):
# Free resources
self._widget.kill_process()

def save_settings(self, plugin_settings, instance_settings):
instance_settings.set_value("command", self._command)
instance_settings.set_value("window_name", self._window_name)

def restore_settings(self, plugin_settings, instance_settings):
self._command = instance_settings.value("command")
self._window_name = instance_settings.value("window_name")
if self._command is not None and self._command != '':
self.add_external_window_widget()
else:
self.trigger_configuration()

def trigger_configuration(self):
# Enable the gear icon and allow to configure the plugin for the program to execute
text, ok = QInputDialog.getText(self._widget, 'rqt_embed_window Settings',
"Qt GUI command to execute (disable splashscreens):",
text=self._command)
if ok:
self._command = text

# Ask if the user wants to use the Window Name instead of the PID to find the window
# Some apps spawn different processes and it's hard to find the window otherwise
text, ok = QInputDialog.getText(self._widget, 'rqt_embed_window Settings',
"If you prefer to find the window by the window name, input it here:",
text=self._window_name)
if ok:
self._window_name = text

# Refresh plugin!
if not self._widget.is_window_empty():
self._widget.remove_widget()

if self._widget.is_process_running():
self._widget.kill_process()

self.add_external_window_widget()
113 changes: 35 additions & 78 deletions src/rqt_embed_window/RqtEmbedWindow.py → src/rqt_embed_window/embed_window_widget.py
100644 → 100755
Original file line number Diff line number Diff line change
Expand Up @@ -2,12 +2,10 @@
import rospy
import time

from qt_gui.plugin import Plugin
from python_qt_binding import loadUi
from python_qt_binding.QtWidgets import QWidget, QInputDialog
from python_qt_binding.QtWidgets import QWidget
from python_qt_binding.QtGui import QWindow
from python_qt_binding.QtCore import Qt
from qt_gui.settings import Settings
from shell_cmd import ShellCmd


Expand Down Expand Up @@ -75,56 +73,48 @@ def wait_for_window_id(pid=None, window_name=None, timeout=5.0):
return window_id


class RqtEmbedWindow(Plugin):
class EmbedWindowWidget(QWidget):
"""
This plugin allows to embed a Qt window into a rqt plugin.
Separated QWidget from plugin
"""

def __init__(self, context):
super(RqtEmbedWindow, self).__init__(context)
self.setObjectName('RqtEmbedWindow')
self._command = ''
self._window_name = ''
def __init__(self, plugin = None):
super(EmbedWindowWidget, self).__init__()
self._external_window_widget = None
self._process = None
self._timeout_to_window_discovery = 20.0
self._plugin = plugin

# Create QWidget
self._widget = QWidget()
# Get path to UI file which is a sibling of this file
# in this example the .ui and .py file are in the same folder
ui_file = os.path.join(os.path.dirname(os.path.realpath(__file__)),
'RqtEmbedWindow.ui')
'EmbedWindow.ui')
# Extend the widget with all attributes and children from UI file
loadUi(ui_file, self._widget)
self._widget.setObjectName('RqtEmbedWindowUi')
# Add widget to the user interface
context.add_widget(self._widget)
loadUi(ui_file, self)
self.setObjectName('EmbedWindowUi')

self.context = context

def add_external_window_widget(self):
def add_external_window_widget(self, command, window_name = "", timeout_to_window_discovery = 20.0):
# The command is prepended with exec so it becomes the shell executing it
# So it effectively has the PID we will look for the window ID
self._process = ShellCmd("exec " + self._command)
self._process = ShellCmd("exec " + command)

# If a window name is provided, it probably means that's the only way to find the window
# so, let's do that first
if self._window_name:
window_id = wait_for_window_id(window_name=self._window_name,
timeout=self._timeout_to_window_discovery)
if window_name:
window_id = wait_for_window_id(window_name=window_name,
timeout=timeout_to_window_discovery)
else:
# Get window ID from PID, we must wait for it to appear
window_id = wait_for_window_id(pid=self._process.get_pid(),
timeout=self._timeout_to_window_discovery)
timeout=timeout_to_window_discovery)
if window_id is None:
rospy.logerr("Could not find window id...")
rospy.logerr("Command was: {} \nWindow name was: '{}'\nStdOut was: {}\nStdErr was: {}".format(self._command,
self._window_name,
rospy.logerr("Command was: {} \nWindow name was: '{}'\nStdOut was: {}\nStdErr was: {}".format(command,
window_name,
self._process.get_stdout(),
self._process.get_stderr()))
self._process.kill()
return
return None

# Create a the window that will contain the program
window = QWindow.fromWinId(window_id)
Expand All @@ -136,55 +126,22 @@ def add_external_window_widget(self):
self._external_window_widget = widget

# Set all margins and spacing to 0 to maximize window usage
self._widget.verticalLayout.setContentsMargins(0, 0, 0, 0)
self._widget.verticalLayout.setSpacing(0)
self.verticalLayout.setContentsMargins(0, 0, 0, 0)
self.verticalLayout.setSpacing(0)

self._widget.verticalLayout.addWidget(self._external_window_widget)
self.verticalLayout.addWidget(self._external_window_widget)

# Give the title (for rqt_gui compisitions) some information
if self.context.serial_number() < 2:
self._widget.setWindowTitle('{} ({})'.format(self._widget.windowTitle(), self._command))
else:
self._widget.setWindowTitle('{} ({}) ({})'.format(self._widget.windowTitle(),
self.context.serial_number(), self._command))

def shutdown_plugin(self):
# Free resources
self._process.kill()

def save_settings(self, plugin_settings, instance_settings):
instance_settings.set_value("command", self._command)
instance_settings.set_value("window_name", self._window_name)

def restore_settings(self, plugin_settings, instance_settings):
self._command = instance_settings.value("command")
self._window_name = instance_settings.value("window_name")
if self._command is not None and self._command != '':
self.add_external_window_widget()
else:
self.trigger_configuration()

def trigger_configuration(self):
# Enable the gear icon and allow to configure the plugin for the program to execute
text, ok = QInputDialog.getText(self._widget, 'RqtEmbedWindow Settings',
"Qt GUI command to execute (disable splashscreens):",
text=self._command)
if ok:
self._command = text

# Ask if the user wants to use the Window Name instead of the PID to find the window
# Some apps spawn different processes and it's hard to find the window otherwise
text, ok = QInputDialog.getText(self._widget, 'RqtEmbedWindow Settings',
"If you prefer to find the window by the window name, input it here:",
text=self._window_name)
if ok:
self._window_name = text

# Refresh plugin!
if self._external_window_widget is not None:
self._widget.verticalLayout.removeWidget(self._external_window_widget)

if self._process is not None and not self._process.is_done():
self._process.kill()

self.add_external_window_widget()
return None

def is_window_empty(self):
return (self._external_window_widget is None)

def remove_widget(self):
self.verticalLayout.removeWidget(self._external_window_widget)

def is_process_running(self):
return (self._process is not None and not self._process.is_done())

def kill_process(self):
if self._process:
self._process.kill()
3 changes: 1 addition & 2 deletions src/rqt_embed_window/shell_cmd.py
100644 → 100755
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,6 @@ def wait_until_done(self):
while not self.is_done():
time.sleep(0.1)


def get_pid(self):
return self.process.pid

Expand All @@ -78,4 +77,4 @@ def kill(self):
print "It printed: " + str(cmd.get_stdout())
else:
print "The command didn't finish succesfully"
print "Its stderr was: " + str(cmd.get_stdout())
print "Its stderr was: " + str(cmd.get_stdout())