diff --git a/README.md b/README.md index cedba6e..cd823cf 100644 --- a/README.md +++ b/README.md @@ -99,11 +99,11 @@ GitHubのProjects機能で開発方針に合わせたタスク管理をしてい - [2nd step - SSL Game Example](https://github.com/SSL-Roots/consai2r2/projects/3) - [3rd step - use ROS2 effectively](https://github.com/SSL-Roots/consai2r2/projects/2) -# Controbution +# Contribution issue, pull request 大歓迎です。 -[CONTORIBUTING.md](./CONTRIBUTING.md)を見てくれると嬉しいです。 +[CONTRIBUTING.md](./CONTRIBUTING.md)を見てくれると嬉しいです。 # Author & Contributors diff --git a/consai2r2_description/CMakeLists.txt b/consai2r2_description/CMakeLists.txt index 9a12a69..d388479 100644 --- a/consai2r2_description/CMakeLists.txt +++ b/consai2r2_description/CMakeLists.txt @@ -13,7 +13,6 @@ endif() # find dependencies find_package(ament_cmake REQUIRED) find_package(rclcpp REQUIRED) -find_package(python_cmake_module REQUIRED) set(_PYTHON_EXECUTABLE "${PYTHON_EXECUTABLE}") @@ -30,7 +29,6 @@ add_executable(${PROJECT_NAME}_node ament_target_dependencies( ${PROJECT_NAME}_node "rclcpp" - "launch_ros" ) diff --git a/consai2r2_description/consai2r2_description/__init__.py b/consai2r2_description/consai2r2_description/__init__.py index 5fdfdf3..4c15d64 100755 --- a/consai2r2_description/consai2r2_description/__init__.py +++ b/consai2r2_description/consai2r2_description/__init__.py @@ -23,8 +23,8 @@ class consai2r2_parameters(): - def __init__(self, node): - param_names = consai2r2_description.parameter.list_parameters(node) - params = consai2r2_description.parameter.get_parameters(node, param_names) + def __init__(self, node, timeout_sec=10.0): + param_names = consai2r2_description.parameter.list_parameters(node, timeout_sec) + params = consai2r2_description.parameter.get_parameters(node, param_names, timeout_sec) for param_name, param_value in params.items(): setattr(self, param_name, param_value) diff --git a/consai2r2_description/consai2r2_description/parameter.py b/consai2r2_description/consai2r2_description/parameter.py index fef2bfc..4149d73 100755 --- a/consai2r2_description/consai2r2_description/parameter.py +++ b/consai2r2_description/consai2r2_description/parameter.py @@ -19,15 +19,15 @@ import rclpy -# https://github.com/ros2/ros2cli/blob/master/ros2param/ros2param/api/__init__.py#L174 -def list_parameters(node): +# https://github.com/ros2/ros2cli/blob/780923c046f8e537e884d18bef33a2338f2d409c/ros2param/ros2param/api/__init__.py#L174 +def list_parameters(node, timeout_sec=10.0): # create client client = node.create_client( ListParameters, 'consai2r2_description/list_parameters') # call as soon as ready - ready = client.wait_for_service(timeout_sec=5.0) + ready = client.wait_for_service(timeout_sec) if not ready: raise RuntimeError('Wait for service timed out') @@ -43,15 +43,15 @@ def list_parameters(node): return response.result.names -# https://github.com/ros2/ros2cli/blob/master/ros2param/ros2param/api/__init__.py#L54 -def get_parameters(node, parameter_names): +# https://github.com/ros2/ros2cli/blob/780923c046f8e537e884d18bef33a2338f2d409c/ros2param/ros2param/api/__init__.py#L122 +def get_parameters(node, parameter_names, timeout_sec=10.0): # create client client = node.create_client( GetParameters, 'consai2r2_description/get_parameters') # call as soon as ready - ready = client.wait_for_service(timeout_sec=5.0) + ready = client.wait_for_service(timeout_sec) if not ready: raise RuntimeError('Wait for service timed out') @@ -68,6 +68,7 @@ def get_parameters(node, parameter_names): return_values = {} + # https://github.com/ros2/ros2cli/blob/780923c046f8e537e884d18bef33a2338f2d409c/ros2param/ros2param/api/__init__.py#L54 for i, pvalue in enumerate(response.values): if pvalue.type == ParameterType.PARAMETER_BOOL: value = pvalue.bool_value diff --git a/consai2r2_examples/launch/joystick_example.launch.py b/consai2r2_examples/launch/joystick_example.launch.py index e033011..396bdd1 100644 --- a/consai2r2_examples/launch/joystick_example.launch.py +++ b/consai2r2_examples/launch/joystick_example.launch.py @@ -18,6 +18,8 @@ # OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN # THE SOFTWARE. +import os +from ament_index_python.packages import get_package_share_directory from launch import LaunchDescription from launch.actions import DeclareLaunchArgument from launch.substitutions import LaunchConfiguration @@ -49,7 +51,9 @@ def generate_launch_description(): start_sender_cmd = Node( package='consai2r2_sender', node_executable='sim_sender', - output='screen' + output='screen', + parameters=[os.path.join(get_package_share_directory( + 'consai2r2_sender'), 'config', 'grsim.yaml')] ) ld = LaunchDescription() diff --git a/consai2r2_examples/package.xml b/consai2r2_examples/package.xml index 2f79e57..5af456e 100644 --- a/consai2r2_examples/package.xml +++ b/consai2r2_examples/package.xml @@ -9,6 +9,7 @@ ament_cmake launch_ros + ament_index_python joy ament_lint_auto ament_lint_common diff --git a/consai2r2_gameviewer/CMakeLists.txt b/consai2r2_gameviewer/CMakeLists.txt new file mode 100644 index 0000000..a0d30db --- /dev/null +++ b/consai2r2_gameviewer/CMakeLists.txt @@ -0,0 +1,23 @@ +cmake_minimum_required(VERSION 3.5) +project(consai2r2_gameviewer) + +# Load ament and all dependencies required for this package +find_package(ament_cmake REQUIRED) +find_package(ament_cmake_python REQUIRED) + +ament_python_install_package(${PROJECT_NAME} + PACKAGE_DIR src/${PROJECT_NAME}) + +install(FILES plugin.xml + DESTINATION share/${PROJECT_NAME} +) + +install(DIRECTORY resource + DESTINATION share/${PROJECT_NAME} +) + +install(PROGRAMS scripts/consai2r2_gameviewer + DESTINATION lib/${PROJECT_NAME} +) + +ament_package() diff --git a/consai2r2_gameviewer/package.xml b/consai2r2_gameviewer/package.xml new file mode 100644 index 0000000..3da52a4 --- /dev/null +++ b/consai2r2_gameviewer/package.xml @@ -0,0 +1,24 @@ + + + + consai2r2_gameviewer + 0.0.0 + consai2r2_gameviewer provides a GUI plugin for displaying vision and referee messages. + akshota + MIT + + ament_cmake + + ament_index_python + python_qt_binding + rclpy + rqt_gui + rqt_gui_py + rqt_py_common + + + + + ament_cmake + + diff --git a/consai2r2_gameviewer/plugin.xml b/consai2r2_gameviewer/plugin.xml new file mode 100644 index 0000000..8217d8d --- /dev/null +++ b/consai2r2_gameviewer/plugin.xml @@ -0,0 +1,17 @@ + + + + A Python GUI plugin for displaying vision and referee messages. + + + + + folder + Plugins related to visualization. + + + image-x-generic + A Python GUI plugin for displaying vision and referee messages. + + + diff --git a/consai2r2_gameviewer/resource/gameviewer_widget.ui b/consai2r2_gameviewer/resource/gameviewer_widget.ui new file mode 100644 index 0000000..1f346fa --- /dev/null +++ b/consai2r2_gameviewer/resource/gameviewer_widget.ui @@ -0,0 +1,19 @@ + + + GameViewerWidget + + + + 0 + 0 + 400 + 300 + + + + GameViewer + + + + + diff --git a/consai2r2_gameviewer/scripts/consai2r2_gameviewer b/consai2r2_gameviewer/scripts/consai2r2_gameviewer new file mode 100644 index 0000000..bcb9058 --- /dev/null +++ b/consai2r2_gameviewer/scripts/consai2r2_gameviewer @@ -0,0 +1,8 @@ +#!/usr/bin/env python3 + +import sys + +from rqt_gui.main import Main + +main = Main() +sys.exit(main.main(sys.argv, standalone='consai2r2_gameviewer.gameviewer.GameViewer')) diff --git a/consai2r2_gameviewer/src/consai2r2_gameviewer/__init__.py b/consai2r2_gameviewer/src/consai2r2_gameviewer/__init__.py new file mode 100644 index 0000000..e69de29 diff --git a/consai2r2_gameviewer/src/consai2r2_gameviewer/gameviewer.py b/consai2r2_gameviewer/src/consai2r2_gameviewer/gameviewer.py new file mode 100644 index 0000000..62826ec --- /dev/null +++ b/consai2r2_gameviewer/src/consai2r2_gameviewer/gameviewer.py @@ -0,0 +1,62 @@ +# Copyright (c) 2019 SSL-Roots +# +# Permission is hereby granted, free of charge, to any person obtaining a copy +# of this software and associated documentation files (the "Software"), to deal +# in the Software without restriction, including without limitation the rights +# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +# copies of the Software, and to permit persons to whom the Software is +# furnished to do so, subject to the following conditions: +# +# The above copyright notice and this permission notice shall be included in +# all copies or substantial portions of the Software. +# +# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL +# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +# THE SOFTWARE. + +from qt_gui.plugin import Plugin +from python_qt_binding.QtCore import QTimer + +from consai2r2_gameviewer.gameviewer_widget import GameViewerWidget + + +class GameViewer(Plugin): + + """ + consai2r2_gameviwer plugin's main class. + """ + + def __init__(self, context): + """ + :param context: plugin context hook to enable adding widgets as a ROS_GUI pane, + ''PluginContext'' + """ + super(GameViewer, self).__init__(context) + self.setObjectName('GameViewer') + + self._context = context + + self._widget = GameViewerWidget() + if context.serial_number() > 1: + self._widget.setWindowTitle( + self._widget.windowTitle() + (' (%d)' % context.serial_number())) + context.add_widget(self._widget) + + self._timer = QTimer() + self._timer.timeout.connect(self._widget.update) + self._timer.start(16) # the argument is msec interval + + def shutdown_plugin(self): + self._timer.stop() + pass + + def save_settings(self, plugin_settings, instance_settings): + pass + + def restore_settings(self, plugin_settings, instance_settings): + pass + diff --git a/consai2r2_gameviewer/src/consai2r2_gameviewer/gameviewer_widget.py b/consai2r2_gameviewer/src/consai2r2_gameviewer/gameviewer_widget.py new file mode 100644 index 0000000..78ede74 --- /dev/null +++ b/consai2r2_gameviewer/src/consai2r2_gameviewer/gameviewer_widget.py @@ -0,0 +1,57 @@ +# Copyright (c) 2019 SSL-Roots +# +# Permission is hereby granted, free of charge, to any person obtaining a copy +# of this software and associated documentation files (the "Software"), to deal +# in the Software without restriction, including without limitation the rights +# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +# copies of the Software, and to permit persons to whom the Software is +# furnished to do so, subject to the following conditions: +# +# The above copyright notice and this permission notice shall be included in +# all copies or substantial portions of the Software. +# +# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL +# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +# THE SOFTWARE. + +import os + +from ament_index_python.resources import get_resource + +from python_qt_binding import loadUi +from python_qt_binding.QtWidgets import QWidget +from python_qt_binding.QtGui import QPainter +from python_qt_binding.QtCore import Qt + +class GameViewerWidget(QWidget): + + """ + Primary widget for the consai2r2_gameviewer plugin. + """ + + def __init__(self): + super(GameViewerWidget, self).__init__() + + pkg_name = 'consai2r2_gameviewer' + _, package_path = get_resource('packages', pkg_name) + ui_file = os.path.join( + package_path, 'share', pkg_name, 'resource', 'gameviewer_widget.ui') + loadUi(ui_file, self) + + self.setObjectName('GameViewerWidget') + + + def paintEvent(self, event): + painter = QPainter(self) + + # Hello world + painter.setBrush(Qt.green) + painter.setPen(Qt.black) + painter.drawRect(self.rect()) + + painter.drawText(self.rect().width()*0.5, self.rect().height()*0.5, "Hello world!") + diff --git a/consai2r2_receiver/include/consai2r2_receiver/multicast.hpp b/consai2r2_receiver/include/consai2r2_receiver/multicast.hpp index 475b182..b62fb64 100644 --- a/consai2r2_receiver/include/consai2r2_receiver/multicast.hpp +++ b/consai2r2_receiver/include/consai2r2_receiver/multicast.hpp @@ -34,22 +34,23 @@ class MulticastReceiver { public: MulticastReceiver(const std::string & ip, const int port) - : endpoint(asio::ip::udp::v4(), port), socket(io_service, endpoint) + : socket_(io_service_, asio::ip::udp::v4()) { asio::ip::address addr = asio::ip::address::from_string(ip); if (!addr.is_multicast()) { - throw std::runtime_error("excpeted multicast address"); + throw std::runtime_error("expected multicast address"); } - socket.set_option(asio::ip::multicast::join_group(addr.to_v4())); - socket.set_option(asio::socket_base::reuse_address(true)); - socket.non_blocking(true); + socket_.set_option(asio::socket_base::reuse_address(true)); + socket_.set_option(asio::ip::multicast::join_group(addr.to_v4())); + socket_.bind(asio::ip::udp::endpoint(asio::ip::udp::v4(), port)); + socket_.non_blocking(true); } size_t receive(std::vector & msg) { boost::system::error_code error; - const size_t received = socket.receive_from(asio::buffer(msg), endpoint, 0, error); + const size_t received = socket_.receive(asio::buffer(msg), 0, error); if (error && error != asio::error::message_size) { throw boost::system::system_error(error); return 0; @@ -57,12 +58,11 @@ class MulticastReceiver return received; } - size_t available() {return socket.available();} + size_t available() {return socket_.available();} private: - asio::io_service io_service; - asio::ip::udp::endpoint endpoint; - asio::ip::udp::socket socket; + asio::io_service io_service_; + asio::ip::udp::socket socket_; }; #endif // CONSAI2R2_RECEIVER__MULTICAST_HPP_ diff --git a/consai2r2_sender/CMakeLists.txt b/consai2r2_sender/CMakeLists.txt index b211072..25813de 100644 --- a/consai2r2_sender/CMakeLists.txt +++ b/consai2r2_sender/CMakeLists.txt @@ -49,6 +49,16 @@ install(TARGETS sim_sender EXPORT export_${PROJECT_NAME} DESTINATION lib/${PROJECT_NAME}) +install(DIRECTORY + config + DESTINATION share/${PROJECT_NAME}/ +) + +install(DIRECTORY + launch + DESTINATION share/${PROJECT_NAME}/ +) + if(BUILD_TESTING) find_package(ament_lint_auto REQUIRED) # the following line skips the linter which checks for copyrights diff --git a/consai2r2_sender/README.md b/consai2r2_sender/README.md new file mode 100644 index 0000000..d83eabd --- /dev/null +++ b/consai2r2_sender/README.md @@ -0,0 +1,53 @@ +# consai2r2_sender + +ロボット(grSim)へ動作司令を送信するパッケージです。 + + + +## ノードの起動方法 + +次のコマンドでsenderが起動します。 + +```sh +ros2 launch consai2r2_sender sender.launch.py +``` + + + + + + + +## 参考ページ +### Google proto files +- grSim + - https://github.com/RoboCup-SSL/grSim/tree/master/src/proto diff --git a/consai2r2_sender/config/grsim.yaml b/consai2r2_sender/config/grsim.yaml new file mode 100644 index 0000000..0c57296 --- /dev/null +++ b/consai2r2_sender/config/grsim.yaml @@ -0,0 +1,4 @@ +consai2r2_sim_sender: + ros__parameters: + grsim_addr: '127.0.0.1' + grsim_port: 20011 diff --git a/consai2r2_sender/launch/sender.launch.py b/consai2r2_sender/launch/sender.launch.py new file mode 100644 index 0000000..2b86d28 --- /dev/null +++ b/consai2r2_sender/launch/sender.launch.py @@ -0,0 +1,39 @@ +# Copyright (c) 2019 SSL-Roots +# +# Permission is hereby granted, free of charge, to any person obtaining a copy +# of this software and associated documentation files (the "Software"), to deal +# in the Software without restriction, including without limitation the rights +# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +# copies of the Software, and to permit persons to whom the Software is +# furnished to do so, subject to the following conditions: +# +# The above copyright notice and this permission notice shall be included in +# all copies or substantial portions of the Software. +# +# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL +# THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN +# THE SOFTWARE. + +import os +from ament_index_python.packages import get_package_share_directory +from launch import LaunchDescription +from launch_ros.actions import Node + + +def generate_launch_description(): + start_sender_cmd = Node( + package='consai2r2_sender', node_executable='sim_sender', + output='screen', + parameters=[os.path.join(get_package_share_directory( + 'consai2r2_sender'), 'config', 'grsim.yaml')] + ) + + ld = LaunchDescription() + + ld.add_action(start_sender_cmd) + + return ld diff --git a/consai2r2_sender/package.xml b/consai2r2_sender/package.xml index bf2df8d..9ebd0d8 100644 --- a/consai2r2_sender/package.xml +++ b/consai2r2_sender/package.xml @@ -15,6 +15,8 @@ protobuf-dev boost + ament_index_python + launch_ros ament_lint_auto ament_lint_common diff --git a/consai2r2_sender/src/sim_sender.cpp b/consai2r2_sender/src/sim_sender.cpp index 2590b09..3145d96 100644 --- a/consai2r2_sender/src/sim_sender.cpp +++ b/consai2r2_sender/src/sim_sender.cpp @@ -63,14 +63,10 @@ class SimSender : public rclcpp::Node SimSender() : Node("consai2r2_sim_sender") { - std::string host; - int port; - this->declare_parameter("grsim_addr", "127.0.0.1"); this->declare_parameter("grsim_port", 20011); - - this->get_parameter("grsim_addr", host); - this->get_parameter("grsim_port", port); + auto host = this->get_parameter("grsim_addr").as_string(); + auto port = this->get_parameter("grsim_port").as_int(); sub_commands_ = this->create_subscription( "robot_commands", 10, std::bind(&SimSender::send_commands, this, std::placeholders::_1));