给SLAM小车添加 手柄遥控 功能 罗技F710和PS4 手柄

news/2025/1/12 20:04:55/

在線測試手柄的好壞:www.gamepad-tester.com

罗技Logetic F710 手柄

手柄正面背后上有D和X拨档,D表示Direct 模式,X表示Xbox 模式,带有陀螺仪数据,一般是游戏中手柄当做鼠标选择或进行打字用。一般遥控机器人的时候没有用到这个信息,所以直接选择D档。

正面小按钮Mode灯亮和不亮的时候,会影响摇杆的数据输出。mode灯亮的模式下,只有右手边摇杆可以在+1和-1之间无极调节(精确到小数点后18位),而其他摇杆只有三种状态输出:+1.0,0.0,-1.0。因此直接选择mode灯不亮的模式,按mode按钮可以来回切换模式。

1. 安装驱动

将硬件设备的信息转为ROS中可以识别的/joy topic 

sudo apt-get install ros-$ROS_DISTRO-joy
sudo apt-get install ros-$ROS_DISTRO-joystick-drivers

http://wiki.ros.org/joy  官方帮助在这里

插入F710 USB接收器到电脑,系统如果识别到, 输入以下指令 ls /dev/input/js*,可以看到一个名为js0的设备接入系统了。

rosrun joy joy_node _dev:=/dev/input/js0

默认是打开 /dev/input/js0 这个设备, 如果是其他设备使用下面设置不同的手柄

rosparam list 

rosparam set joy_node/dev "/dev/input/js1"

rostopic echo /joy 然后拨弄遥控手柄,就可以查看信息交互信息了。上面的mode 模式选择的无极调节和非无极调节就是通过这个debug看出来的。 后文的launch文件的数字就是这个表里面来的。

罗技F710 手柄 摇杆 键值对应关系(注意此时mode灯处于 熄灭状态)
axis012345
左(左右)左(前后)右(左右)右(前后)十字键左右十字键前后

罗技F710 按钮 键值对应关系表
button01234567891011
X(左)A(下)B(右)Y(上)LTRTLBRBBackStartL3R3

2. 订阅joy并发布cmd_vel 

 关于这个有两个系统package可以用,第一个代码简单比较直接,第二个用的pimpl实现代码规范但略微复杂。这里选择turtlebot的

turtlebot_teleop - ROS Wiki

teleop_twist_joy - ROS Wiki

cd ~/catkin_ws/src
git clone https://github.com/turtlebot/turtlebot.git

官方程序源码地址:    turtlebot/turtlebot_joy.cpp at melodic · turtlebot/turtlebot (github.com)  

官方的程序两个小问题,第一个vel的topic名字没有设成变量,默认为cmd_vel,不利于后面设置多输入速度参数配置,要修改的话必须改源码。 joy名字也是如此 。

第二个如果速度不合理,(比如一开始scale_angular和scale_linear设置低了)需要先结束launch程序找到并修改launch文件,再重启程序。手柄上还空有很多键,可以加个步进按钮调整(这里利用手柄上的X和Y键,每次按下按钮速度就上下调整0.05)和Reset 按钮(Button序号5,查表知道为RT),按下此按钮,小车速度复位。也即为小车设置了低速档和高速档。

#include <ros/ros.h>
#include <geometry_msgs/Twist.h>
#include <sensor_msgs/Joy.h>
#include "boost/thread/mutex.hpp"
#include "boost/thread/thread.hpp"
#include "ros/console.h"
#include <iostream>
using namespace std;class Teleop
{
public:Teleop();private:void joyCallback(const sensor_msgs::Joy::ConstPtr& joy);void publish();ros::NodeHandle ph_, nh_;int linear_, angular_, deadman_axis_,l_up,l_down;double l_scale_, a_scale_;ros::Publisher vel_pub_;ros::Subscriber joy_sub_;geometry_msgs::Twist last_published_;boost::mutex publish_mutex_;bool deadman_pressed_;bool zero_twist_published_;ros::Timer timer_;int reset_;double l_reset_, a_reset_;std::string vel_name_;std::string joy_name_;};Teleop::Teleop():ph_("~"),linear_(1),angular_(2),deadman_axis_(4),l_up(3),l_down(1),reset_(5),l_scale_(0.2),a_scale_(0.4),l_reset_(0.2),a_reset_(0.4)	
{ph_.param("axis_linear", linear_, linear_);ph_.param("axis_angular", angular_, angular_);ph_.param("axis_deadman", deadman_axis_, deadman_axis_);ph_.param("scale_angular", a_scale_, a_scale_);ph_.param("scale_linear", l_scale_, l_scale_);ph_.param("reset_btn", reset_, reset_);ph_.param("angular_reset", a_reset_, a_reset_);ph_.param("linear_reset", l_reset_, l_reset_);ph_.param<std::string>("vel_name", vel_name_, std::string("/turtle1/cmd_vel"));ph_.param<std::string>("joy_name", joy_name_, std::string("/joy"));deadman_pressed_ = false;zero_twist_published_ = false;cout<<"vel topic name: "<<vel_name_<<endl;cout<<"joy topic name: "<<joy_name_<<endl;vel_pub_ = nh_.advertise<geometry_msgs::Twist>(vel_name_, 1, true);joy_sub_ = nh_.subscribe<sensor_msgs::Joy>(joy_name_, 10, &Teleop::joyCallback, this);timer_ = nh_.createTimer(ros::Duration(0.1), boost::bind(&Teleop::publish, this));
}void Teleop::joyCallback(const sensor_msgs::Joy::ConstPtr& joy)
{geometry_msgs::Twist vel;if(joy->axes[angular_])a_scale_ = a_scale_ + 0.05 * joy->buttons[l_up] - 0.05 * joy->buttons[l_down];if(joy->axes[linear_])l_scale_ = l_scale_ + 0.05 * joy->buttons[l_up] - 0.05 * joy->buttons[l_down];if(joy->buttons[reset_]){l_scale_ = l_reset_;a_scale_ = a_reset_;}vel.angular.z = a_scale_*joy->axes[angular_];vel.linear.x = l_scale_*joy->axes[linear_];last_published_ = vel;cout<<"cmd_cel value: x "<<vel.linear.x<< " angular "<<vel.angular.z<<endl;deadman_pressed_ = joy->buttons[deadman_axis_];
}void Teleop::publish()
{boost::mutex::scoped_lock lock(publish_mutex_);if (deadman_pressed_){vel_pub_.publish(last_published_);zero_twist_published_=false;}else if(!deadman_pressed_ && !zero_twist_published_){vel_pub_.publish(*new geometry_msgs::Twist());zero_twist_published_=true;}
}int main(int argc, char** argv)
{ros::init(argc, argv, "joy_car");Teleop robot_teleop;ros::spin();
}

在自己的一个ROS package工作目录下建一个logetic_joy_teleop.launch,内容如下。

直线速度最大值(scale_linear)和角速度最大值(scale_angular),油门离合键(axis_deadman 4)、前进后退轴(axis_linear,1 ,查文章前面的对应表知道对应手柄左摇杆的前后位)、左右转轴(axis_angular,2,对应手柄右摇杆的左右位)

<launch><node pkg="turtlebot_teleop" type="turtlebot_teleop_joy" name="turtlebot_teleop_joy" output="screen" clear_params="true"><param name="scale_angular" value="0.2"/><param name="scale_linear" value="0.2"/><param name="linear_reset" value="0.2"/><param name="angular_reset" value="0.4"/><!-- Left up down --><param name="axis_linear" value="1"/><!-- Right left right --><param name="axis_angular" value="2"/><!-- L1 --><param name="axis_deadman" value="4"/><!-- R1 --><param name="reset_btn" value="5"/><param name="vel_name" value="/joy_cmd_vel" type="string" /><param name="joy_name" value="/joy" type="string" /></node><node pkg="joy" type="joy_node" name="joystick"><param name="dev" value="/dev/input/js0" type="string" /></node></launch>

PS4 手柄长按share和PS键开启蓝牙配对模式,此时手柄灯闪烁白光。配对成功后白灯常亮。

PS4 手柄  摇杆 键值对应关系
axis012345

(左右)

(前后)

L2

默认为1, 按下递减到-1  

(左右)

(前后)

R2

默认为1, 按下递减到-1  

PS4 手柄 按键 键值对应关系
button01234567891011121314

×

(下)

(右)

(左)

△ (上)

L1

R1

share

options

PS 键L3R3十字左

十字右

十字上

十字下

husky/teleop_ps4.yaml at noetic-devel · husky/husky (github.com)

# Teleop configuration for PS4 joystick using the x-pad configuration.
# Left thumb-stick up/down for velocity, left/right for twist
# Left shoulder button for enable
# Right shoulder button for enable-turbo
#
#          L1                                       R1
#          L2                                       R2
#       _=====_                                  _=====_
#      / _____ \                                / _____ \
#    +.-'_____'-.------------------------------.-'_____'-.+
#   /   |     |  '.        S O N Y           .'  |  _  |   \
#  / ___| /|\ |___ \                        / ___| /_\ |___ \      (Y)
# / |      |      | ;                      ; | _         _ ||
# | | <---   ---> | |                      | ||_|       (_)||  (X)     (B)
# | |___   |   ___| ;                      ; |___       ___||
# |\    | \|/ |    /  _      ____      _   \    | (X) |    /|      (A)
# | \   |_____|  .','" "',  (_PS_)  ,'" "', '.  |_____|  .' |
# |  '-.______.-' /       \        /       \  '-._____.-'   |
# |               |  LJ   |--------|  RJ   |                |
# |              /\       /        \       /\               |
# |             /  '.___.'          '.___.'  \              |
# |            /                              \             |
#  \          /                                \           /
#   \________/                                  \_________/
#
#          ^ x
#          |
#          |
#  y <-----+      Accelerometer axes
#           \
#            \
#             > z (out)
#
# BUTTON         Value
#   L1             4
#   L2             6
#   R1             5
#   R2             7
#   A              1
#   B              2
#   X              0
#   Y              3
#
#    AXIS        Value
# Left Horiz.      0
# Left Vert.       1
# Right Horiz.     2
# Right Vert.      5
#    L2            3
#    R2            4
# D-pad Horiz.     9
# D-pad Vert.      10
# Accel x          7
# Accel y          6
# Accel z          8teleop_twist_joy:axis_linear: 1scale_linear: 0.4scale_linear_turbo: 2.0axis_angular: 0scale_angular: 1.4enable_button: 4enable_turbo_button: 5
joy_node:deadzone: 0.1autorepeat_rate: 20dev: /dev/input/ps4

一个简要的ROS下手柄控制程序发布cmd_vel

#include <ros/ros.h>
#include<queue>
#include<sensor_msgs/Joy.h>
#include<geometry_msgs/Twist.h>struct Velocity{double linear = 0;double angle = 0;
};Velocity joy_vel;
static bool tag = false;
const int max_vel = 1;void joy_callback(const sensor_msgs::JoyConstPtr& msg)
{joy_vel.linear = msg->axes[1] * max_vel*1.0;joy_vel.angle = msg->axes[2] * max_vel*1.5; 
}int main(int argc,char** argv)
{ros::init(argc,argv,"joy_control_node");ros::NodeHandle n;ros::NodeHandle p;ros::Subscriber joy_sub;ros::Publisher cmd_pub;joy_sub = n.subscribe("/joy",1,&joy_callback);cmd_pub = p.advertise<geometry_msgs::Twist>("/cmd_vel",1);geometry_msgs::Twist car_vel;ROS_INFO("start to joy control");ros::Rate r(100);while(ros::ok()){car_vel.linear.x = joy_vel.linear;car_vel.linear.y = 0.0;car_vel.linear.z = 0.0;car_vel.angular.z = joy_vel.angle;car_vel.angular.y = 0.0;car_vel.angular.x = 0.0;cmd_pub.publish(car_vel);ros::spinOnce();r.sleep();}ros::shutdown();return 0;}


http://www.ppmy.cn/news/262025.html

相关文章

爷青回|用Python制作【打砖块游戏】相信每一个90后都在洛基亚手机上玩过吧~

&#x1f514; B站主页&#xff1a;https://space.bilibili.com/1707990930&#x1f4e3; 欢迎&#x1f389;点赞&#x1f44d;收藏&#x1f31f;评论&#x1f4dd;如有错误请指正!&#x1f4bb; Python&Java领域博主&#x1f601; 你们的支持是我最大的动力 大家好&#x…

神经网络学习率指数衰减ExponentialDecay策略的参数含义与使用方法详解

本文介绍在tensorflow库中&#xff0c;用于动态调整神经网络的学习率的一种方法——指数衰减ExponentialDecay()策略的参数含义及其具体用法。 在进行神经网络训练时&#xff0c;我们经常需要用到动态变化的学习率&#xff0c;其中指数衰减ExponentialDecay()策略是我们常用的一…

Pandas进阶修炼120题

文章目录 Pandas进阶修炼120题第一期 Pandas基础1.将下面的字典创建为DataFrame2.提取含有字符串"Python"的行3.输出df的所有列名4.修改第二列列名为popularity5.统计grammer列中每种编程语言出现的次数6.将空值用上下值的平均值填充通常方法&#xff08;直接写逻辑&…

麓言信息 交互设计师主要做什么?需要会什么?

随着时代的发展&#xff0c;当前越来越多的企业开始注意到用户界面设计的重要性&#xff0c;所以行业对UI设计师的个人能力要求在不断地提升&#xff0c;除了要具有良好的设计能力、创新设计意识之外&#xff0c;还要懂交互设计与用户体验设计。一时间&#xff0c;交互设计师也…

Spring Boot 用户管理的例子

当我们使用Spring Boot框架时&#xff0c;它可以自动为我们配置很多非常常用的依赖&#xff0c;其中就包括MyBatis和IOC容器。下面给出一个使用Spring Boot、MyBatis和IOC技术创建用户管理网站的简单示例&#xff1a; 1. 确保在您的系统中已经安装了Maven和Java 1.8。 2. 创建…

地震勘探基础(十三)之地震资料解释

地震资料解释 地震资料解释&#xff08;seismic interpretation&#xff09;就是把经过采集和计算机处理后的地震数据转变为地质信息的过程。也就是由已知实际观测的地震数据反演地下地质特征的过程&#xff0c;因此地震资料解释也可称为地震反演。 根据地震资料类型不同&…

管理项目-加载菜单

人事管理项目-加载菜单 引入ElementUI菜单1&#xff0e;菜单模板 引入ElementUI 数据成功访问后&#xff0c;引入ElementUI组件对数据进行渲染&#xff0c;首先引入其依赖&#xff1a; npm install element-ui -S 依赖添加成功后&#xff0c;接着在main.js中引入ElementUI&a…

4k动漫壁纸26

在这里插入图片描述