通过外部程序获得Moveit-Rviz的MotionPlanning面板相同的操控能力

本文主要是介绍通过外部程序获得Moveit-Rviz的MotionPlanning面板相同的操控能力,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在Moveit-Rviz启动之后,在机器人的末端会显示一个Interactive Marker用来指示目标方位,可以通过鼠标来移动和旋转该Marker,操作时会有一个虚拟的机器人随之运动。当按下MotionPlanning面板中的Plan按钮便会显示从当前方位到目标方位的一个运动轨迹。当按下Execute按钮后机器人就会实际运动到目标方位。

Moveit在rviz中提供的操控Topic

当勾选MotionPlanning面板中的"Allow External Comm."选项的时候,在ROS上就会出现如下的几个Topic:

  • /rviz/moveit/plan : 和Rviz中MotionPlanning面板提供的Plan按钮功能一样,该topic的消息类型为std_msgs/Empty,发送一个这样的消息到该Topic就相当与按下了Plan按钮。
  • /rviz/moveit/execute : 和Rviz中MotionPlanning面板提供的Execute按钮功能一样
  • /rviz/moveit/select_planning_group : 用于选择你想要控制的Planning Group的Topic。其消息类型为std_msgs/String。你可以在命令终端下选择Planning Group,例如
    rostopic pub /rviz/moveit/select_planning_group std_msgs/String "hand"
    
  • /rviz/moveit/update_{goal,start}_state : 和MotionPlanning面板中"State State"和"Goal State"下选框中的"current"是同样的功能。
  • /rviz/moveit/update_custom_{goal,start}_state : 该Topic用于给机器人设置一个指定的状态,其消息类型为moveit_msgs/RobotState。
  • /rviz/moveit/move_marker/goal_#{link_name} : 指定交互式marker的方位的Topic。link_name是Planning Group中的末端连杆的名称,例如"/rviz/moveit/move_marker/goal_panda_link8。其中"panda_link8"为panda机器人本体的最后一个连杆。该Topic的消息类型为geometry_msgs/PoseStamped。

另外,Moveit发布如下的Topic

  • /rviz_moveit_motion_planning_display/robot_interactive_interactive_marker_topic/update : 该Topic会发布经过确认的有效的IMarker方位。我们可以用来更新IMarker的实际goalPose。
  • /execute_trajectory/status : 当moveit通过Execute执行实际的轨迹运动时,就会通过该Topic发布执行的状态。因此当我们进行实际的轨迹运动时就可以通过该Topic来检查完成结果。该Topic类型为actionlib_msgs/GoalStatusArray。

使用键盘的简单控制

有了上面的Topic,我们便可以通过编写程序来操控rviz中的机器人,最主要的就是可以自己编写程序控制机器人上的Interactive Marker,例如使用键盘和手柄控制,而非只是简单地在屏幕上进行非精确地拖动。

在这个例子中,我们使用adsw按键来控制Marker在XY平面的移动,通过上下箭头按键控制X方向上的移动。

#!/usr/bin/env python
# coding: UTF-8import rospy
import sys, select, termios, ttyfrom moveit_msgs.msg import RobotState
from geometry_msgs.msg import PoseStamped
from visualization_msgs.msg import InteractiveMarkerUpdate
import moveit_commanderclass IMarkerKeyboard:def __init__(self):self.pre_pose= PoseStamped()self.planning_groups_tips = {}self.ttySettings=termios.tcgetattr(sys.stdin)# 获得机器人的当前实际位置group_name = "arm"  # 这是对机器人进行moveit配置定义的关节组名称self.group = moveit_commander.MoveGroupCommander(group_name)self.robot = moveit_commander.RobotCommander()self.curPose=self.group.get_current_pose()self.initState= self.robot.get_current_state()# 创建相关self.goalState_pub = rospy.Publisher("/rviz/moveit/update_custom_goal_state",RobotState,queue_size=5)self.marker_pub = rospy.Publisher("/rviz/moveit/move_marker/goal_link_6",PoseStamped,queue_size=5)self.update_sub = rospy.Subscriber("/rviz_moveit_motion_planning_display/robot_interaction_interactive_marker_topic/update",InteractiveMarkerUpdate,self.updateGoal)rospy.sleep(2)  # 确保rviz能收到消息# 将IMarker移动到与实际位置一致self.goalState_pub.publish(self.initState)rospy.sleep(1)# 回调函数通过接收update Topic来更新Marker的目标位置def updateGoal(self, msg):if len(msg.poses) > 0:self.curPose.pose.position = msg.poses[0].pose.positionself.curPose.pose.orientation = msg.poses[0].pose.orientationdef getKey(self):tty.setraw(sys.stdin.fileno())select.select([sys.stdin],[],[],0)key= sys.stdin.read(1)termios.tcsetattr(sys.stdin, termios.TCSADRAIN, self.ttySettings)return keydef run(self):print("wait key...")while(True) :key = self.getKey()if key=='-':goalPose=self.curPosegoalPose.pose.position.z -=0.01self.marker_pub.publish(goalPose)elif key=='=':goalPose=self.curPosegoalPose.pose.position.z +=0.01self.marker_pub.publish(goalPose)elif key=='a':goalPose=self.curPosegoalPose.pose.position.x -=0.01self.marker_pub.publish(goalPose)elif key == 'd':goalPose =self.curPosegoalPose.pose.position.x += 0.01self.marker_pub.publish(goalPose)elif key == 's':goalPose =self.curPosegoalPose.pose.position.y -= 0.01self.marker_pub.publish(goalPose)elif key == 'w':goalPose =self.curPosegoalPose.pose.position.y += 0.01self.marker_pub.publish(goalPose)if(key == '\x03'):  # Ctrl+Cbreakdef main():rospy.init_node('marker_test', anonymous=True)app=IMarkerKeyboard()app.run()returnif __name__ == '__main__':main()

使用手柄的简单控制

在手柄控制中,使用如下按键或摇杆进行控制:

  • 左摇杆上下: X方向粗控制
  • 左摇杆左右: Y方向粗控制
  • LEFT-RIGHT: X方向细控制
  • UP-DOWN: Y方向细控制
  • 右摇杆上下: Z方向粗控制
  • X-Y: Z方向细控制

这里并没有对旋转进行控制,其实实现也是很简单的。我们可以复用上面的按键或摇杆,设置移动和旋转两种模式,通过一个按钮在两种模式间进行切换即可。

import rospy
from moveit_msgs.msg import RobotState
from geometry_msgs.msg import PoseStamped
from visualization_msgs.msg import InteractiveMarkerUpdate
from sensor_msgs.msg import  Joy
import moveit_commanderclass IMarkerJoystick:def __init__(self):self.pre_pose= PoseStamped()self.planning_groups_tips = {}self.ttySettings=termios.tcgetattr(sys.stdin)# 获得机器人的当前实际位置group_name = "arm"  # 这是对机器人进行moveit配置定义的关节组名称self.group = moveit_commander.MoveGroupCommander(group_name)self.robot = moveit_commander.RobotCommander()self.curPose=self.group.get_current_pose()self.initState= self.robot.get_current_state()# 创建相关self.goalState_pub = rospy.Publisher("/rviz/moveit/update_custom_goal_state",RobotState,queue_size=5)self.marker_pub = rospy.Publisher("/rviz/moveit/move_marker/goal_link_6",PoseStamped,queue_size=5)self.update_sub = rospy.Subscriber("/rviz_moveit_motion_planning_display/robot_interaction_interactive_marker_topic/update",InteractiveMarkerUpdate,self.updateGoal)self.joy_sub = rospy.Subscriber("/joy", Joy, self.joyCB, queue_size=1)rospy.sleep(2)  # 确保rviz能收到消息# 将IMarker移动到与实际位置一致self.goalState_pub.publish(self.initState)rospy.sleep(1)# 回调函数通过接收update Topic来更新Marker的目标位置def updateGoal(self, msg):if len(msg.poses) > 0:self.curPose.pose.position = msg.poses[0].pose.positionself.curPose.pose.orientation = msg.poses[0].pose.orientationdef joyCB(self,msg):goalPose = self.curPosestatus = XBoxStatus(msg)print(status.select, status.start)print(status.circle, status.center, status.cross, status.square)# 粗调if status.left_analog_x <=-0.55 :goalPose.pose.position.x -=0.001elif status.left_analog_x >= 0.55 :goalPose.pose.position.x +=0.001if status.left_analog_y <= -0.55:goalPose.pose.position.y -= 0.001elif status.left_analog_y >= 0.55:goalPose.pose.position.y += 0.001if status.right_analog_y<=-0.55:goalPose.pose.position.z -=0.001if status.right_analog_y>=0.55:goalPose.pose.position.z += 0.001# 细调if status.left:goalPose.pose.position.x -=0.001if status.right:goalPose.pose.position.x +=0.001if status.up:goalPose.pose.position.y -=0.001if status.down:goalPose.pose.position.y +=0.001if status.square:goalPose.pose.position.z -=0.001if status.triangle:goalPose.pose.position.z +=0.001self.marker_pub.publish(goalPose)

这篇关于通过外部程序获得Moveit-Rviz的MotionPlanning面板相同的操控能力的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/443381

相关文章

python编写朋克风格的天气查询程序

《python编写朋克风格的天气查询程序》这篇文章主要为大家详细介绍了一个基于Python的桌面应用程序,使用了tkinter库来创建图形用户界面并通过requests库调用Open-MeteoAPI... 目录工具介绍工具使用说明python脚本内容如何运行脚本工具介绍这个天气查询工具是一个基于 Pyt

Ubuntu设置程序开机自启动的操作步骤

《Ubuntu设置程序开机自启动的操作步骤》在部署程序到边缘端时,我们总希望可以通电即启动我们写好的程序,本篇博客用以记录如何在ubuntu开机执行某条命令或者某个可执行程序,需要的朋友可以参考下... 目录1、概述2、图形界面设置3、设置为Systemd服务1、概述测试环境:Ubuntu22.04 带图

Java -jar命令如何运行外部依赖JAR包

《Java-jar命令如何运行外部依赖JAR包》在Java应用部署中,java-jar命令是启动可执行JAR包的标准方式,但当应用需要依赖外部JAR文件时,直接使用java-jar会面临类加载困... 目录引言:外部依赖JAR的必要性一、问题本质:类加载机制的限制1. Java -jar的默认行为2. 类加

java -jar命令运行 jar包时运行外部依赖jar包的场景分析

《java-jar命令运行jar包时运行外部依赖jar包的场景分析》:本文主要介绍java-jar命令运行jar包时运行外部依赖jar包的场景分析,本文给大家介绍的非常详细,对大家的学习或工作... 目录Java -jar命令运行 jar包时如何运行外部依赖jar包场景:解决:方法一、启动参数添加: -Xb

Python程序打包exe,单文件和多文件方式

《Python程序打包exe,单文件和多文件方式》:本文主要介绍Python程序打包exe,单文件和多文件方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录python 脚本打成exe文件安装Pyinstaller准备一个ico图标打包方式一(适用于文件较少的程

Python程序的文件头部声明小结

《Python程序的文件头部声明小结》在Python文件的顶部声明编码通常是必须的,尤其是在处理非ASCII字符时,下面就来介绍一下两种头部文件声明,具有一定的参考价值,感兴趣的可以了解一下... 目录一、# coding=utf-8二、#!/usr/bin/env python三、运行Python程序四、

无法启动此程序因为计算机丢失api-ms-win-core-path-l1-1-0.dll修复方案

《无法启动此程序因为计算机丢失api-ms-win-core-path-l1-1-0.dll修复方案》:本文主要介绍了无法启动此程序,详细内容请阅读本文,希望能对你有所帮助... 在计算机使用过程中,我们经常会遇到一些错误提示,其中之一就是"api-ms-win-core-path-l1-1-0.dll丢失

SpringBoot后端实现小程序微信登录功能实现

《SpringBoot后端实现小程序微信登录功能实现》微信小程序登录是开发者通过微信提供的身份验证机制,获取用户唯一标识(openid)和会话密钥(session_key)的过程,这篇文章给大家介绍S... 目录SpringBoot实现微信小程序登录简介SpringBoot后端实现微信登录SpringBoo

uniapp小程序中实现无缝衔接滚动效果代码示例

《uniapp小程序中实现无缝衔接滚动效果代码示例》:本文主要介绍uniapp小程序中实现无缝衔接滚动效果的相关资料,该方法可以实现滚动内容中字的不同的颜色更改,并且可以根据需要进行艺术化更改和自... 组件滚动通知只能实现简单的滚动效果,不能实现滚动内容中的字进行不同颜色的更改,下面实现一个无缝衔接的滚动

C#通过进程调用外部应用的实现示例

《C#通过进程调用外部应用的实现示例》本文主要介绍了C#通过进程调用外部应用的实现示例,以WINFORM应用程序为例,在C#应用程序中调用PYTHON程序,具有一定的参考价值,感兴趣的可以了解一下... 目录窗口程序类进程信息类 系统设置类 以WINFORM应用程序为例,在C#应用程序中调用python程序