给小车添加 手柄遥控 功能 罗技F710和PS4 手柄
Posted yaked
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了给小车添加 手柄遥控 功能 罗技F710和PS4 手柄相关的知识,希望对你有一定的参考价值。
罗技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/input/js0 这个设备, 如果是其他设备使用下面设置不同的手柄
rosparam list
rosparam set joy_node/dev "/dev/input/js1"
用 rostopic echo /joy 然后拨弄遥控手柄,就可以查看信息交互信息了。上面的mode 模式选择的无极调节和非无极调节就是通过这个debug看出来的。 后文的launch文件的数字就是这个表里面来的。
axis | 0 | 1 | 2 | 3 | 4 | 5 |
左(左右) | 左(前后) | 右(左右) | 右(前后) | 十字键左右 | 十字键前后 |
button | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 |
X(左) | A(下) | B(右) | Y(上) | LT | RT | LB | RB | Back | Start | L3 | R3 |
2. 订阅joy并发布cmd_vel
关于这个有两个系统package可以用,第一个代码简单比较直接,第二个用的pimpl实现代码规范但略微复杂。这里选择turtlebot的
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键开启蓝牙配对模式,此时手柄灯闪烁白光。配对成功后白灯常亮。
axis | 0 | 1 | 2 | 3 | 4 | 5 |
左 (左右) | 左 (前后) | L2 默认为1, 按下递减到-1 | 右 (左右) | 右 (前后) | R2 默认为1, 按下递减到-1 |
button | 0 | 1 | 2 | 3 | 4 | 5 | 6 | 7 | 8 | 9 | 10 | 11 | 12 | 13 | 14 |
× (下) | ○ (右) | □ (左) | △ (上) | L1 | R1 | share | options | PS 键 | L3 | R3 | 十字左 |
十字右 | 十字上 | 十字下 |
以上是关于给小车添加 手柄遥控 功能 罗技F710和PS4 手柄的主要内容,如果未能解决你的问题,请参考以下文章
给PS4装上Linux,然后用Xbox手柄打任天堂游戏?网友:这场面我没见过