mirror of
https://github.com/ArduPilot/ardupilot
synced 2025-01-15 13:18:28 -04:00
470 lines
16 KiB
C++
470 lines
16 KiB
C++
#include "Sub.h"
|
|
|
|
// Functions that will handle joystick/gamepad input
|
|
// ----------------------------------------------------------------------------
|
|
|
|
// Anonymous namespace to hold variables used only in this file
|
|
namespace {
|
|
int16_t mode_switch_pwm = 1100;
|
|
float cam_tilt = 1500.0;
|
|
int16_t lights1 = 1100;
|
|
int16_t lights2 = 1100;
|
|
int16_t rollTrim = 0;
|
|
int16_t pitchTrim = 0;
|
|
int16_t zTrim = 0;
|
|
int16_t xTrim = 0;
|
|
int16_t yTrim = 0;
|
|
int16_t video_switch = 1100;
|
|
int16_t x_last, y_last, z_last;
|
|
uint16_t buttons_prev;
|
|
float gain;
|
|
bool toggle_mode = true;
|
|
}
|
|
|
|
void Sub::init_joystick()
|
|
{
|
|
default_js_buttons();
|
|
|
|
set_mode((control_mode_t)flight_modes[0].get(), MODE_REASON_TX_COMMAND); // Initialize flight mode
|
|
|
|
if (g.numGainSettings < 1) {
|
|
g.numGainSettings.set_and_save(1);
|
|
}
|
|
|
|
if (g.numGainSettings == 1 || (g.gain_default < g.maxGain + 0.01 && g.gain_default > g.minGain - 0.01)) {
|
|
gain = constrain_float(g.gain_default, g.minGain, g.maxGain); // Use default gain parameter
|
|
} else {
|
|
// Use setting closest to average of minGain and maxGain
|
|
gain = g.minGain + (g.numGainSettings/2 - 1) * (g.maxGain - g.minGain) / (g.numGainSettings - 1);
|
|
}
|
|
|
|
gain = constrain_float(gain, 0.1, 1.0);
|
|
}
|
|
|
|
void Sub::transform_manual_control_to_rc_override(int16_t x, int16_t y, int16_t z, int16_t r, uint16_t buttons)
|
|
{
|
|
|
|
int16_t channels[11];
|
|
|
|
uint32_t tnow_ms = millis();
|
|
|
|
float rpyScale = 0.4*gain; // Scale -1000-1000 to -400-400 with gain
|
|
float throttleScale = 0.8*gain*g.throttle_gain; // Scale 0-1000 to 0-800 times gain
|
|
int16_t rpyCenter = 1500;
|
|
int16_t throttleBase = 1500-500*throttleScale;
|
|
|
|
bool shift = false;
|
|
static uint32_t buttonDebounce;
|
|
|
|
// Debouncing timer
|
|
if (tnow_ms - buttonDebounce > 100) {
|
|
// Detect if any shift button is pressed
|
|
for (uint8_t i = 0 ; i < 16 ; i++) {
|
|
if ((buttons & (1 << i)) && get_button(i)->function() == JSButton::button_function_t::k_shift) {
|
|
shift = true;
|
|
}
|
|
}
|
|
|
|
// Act if button is pressed
|
|
// Only act upon pressing button and ignore holding. This provides compatibility with Taranis as joystick.
|
|
for (uint8_t i = 0 ; i < 16 ; i++) {
|
|
if ((buttons & (1 << i))) {
|
|
handle_jsbutton_press(i,shift,(buttons_prev & (1 << i)));
|
|
buttonDebounce = tnow_ms;
|
|
}
|
|
}
|
|
|
|
buttons_prev = buttons;
|
|
}
|
|
|
|
// Set channels to override
|
|
channels[0] = 1500 + pitchTrim; // pitch
|
|
channels[1] = 1500 + rollTrim; // roll
|
|
channels[2] = constrain_int16((z+zTrim)*throttleScale+throttleBase,1100,1900); // throttle
|
|
channels[3] = constrain_int16(r*rpyScale+rpyCenter,1100,1900); // yaw
|
|
channels[4] = mode_switch_pwm; // for testing only
|
|
channels[5] = constrain_int16((x+xTrim)*rpyScale+rpyCenter,1100,1900); // forward for ROV
|
|
channels[6] = constrain_int16((y+yTrim)*rpyScale+rpyCenter,1100,1900); // lateral for ROV
|
|
channels[7] = cam_tilt; // camera tilt
|
|
channels[8] = lights1; // lights 1
|
|
channels[9] = lights2; // lights 2
|
|
channels[10] = video_switch; // video switch
|
|
|
|
// Store old x, y, z values for use in input hold logic
|
|
x_last = x;
|
|
y_last = y;
|
|
z_last = z;
|
|
|
|
// record that rc are overwritten so we can trigger a failsafe if we lose contact with groundstation
|
|
failsafe.rc_override_active = hal.rcin->set_overrides(channels, 10);
|
|
}
|
|
|
|
void Sub::handle_jsbutton_press(uint8_t button, bool shift, bool held)
|
|
{
|
|
// For attempts to change control mode
|
|
control_mode_t next_mode = control_mode;
|
|
uint16_t next_mode_switch_pwm = mode_switch_pwm;
|
|
|
|
// Act based on the function assigned to this button
|
|
switch (get_button(button)->function(shift)) {
|
|
case JSButton::button_function_t::k_arm_toggle:
|
|
if (motors.armed()) {
|
|
init_disarm_motors();
|
|
} else {
|
|
init_arm_motors(true);
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_arm:
|
|
init_arm_motors(true);
|
|
break;
|
|
case JSButton::button_function_t::k_disarm:
|
|
init_disarm_motors();
|
|
break;
|
|
case JSButton::button_function_t::k_mode_toggle:
|
|
if (!held) {
|
|
next_mode = (control_mode_t)flight_modes[toggle_mode?1:0].get();
|
|
next_mode_switch_pwm = toggle_mode?1300:1100;
|
|
toggle_mode = !toggle_mode;
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_mode_1:
|
|
next_mode = (control_mode_t)flight_modes[0].get();
|
|
next_mode_switch_pwm = 1100;
|
|
toggle_mode = true;
|
|
break;
|
|
case JSButton::button_function_t::k_mode_2:
|
|
next_mode = (control_mode_t)flight_modes[1].get();
|
|
next_mode_switch_pwm = 1300;
|
|
toggle_mode = false;
|
|
break;
|
|
case JSButton::button_function_t::k_mode_3:
|
|
next_mode = (control_mode_t)flight_modes[2].get();
|
|
next_mode_switch_pwm = 1420;
|
|
toggle_mode = false;
|
|
break;
|
|
case JSButton::button_function_t::k_mode_4:
|
|
next_mode = (control_mode_t)flight_modes[3].get();
|
|
next_mode_switch_pwm = 1550;
|
|
toggle_mode = false;
|
|
break;
|
|
case JSButton::button_function_t::k_mode_5:
|
|
next_mode = (control_mode_t)flight_modes[4].get();
|
|
next_mode_switch_pwm = 1690;
|
|
toggle_mode = false;
|
|
break;
|
|
case JSButton::button_function_t::k_mode_6:
|
|
next_mode = (control_mode_t)flight_modes[5].get();
|
|
next_mode_switch_pwm = 1900;
|
|
toggle_mode = false;
|
|
break;
|
|
case JSButton::button_function_t::k_mount_center:
|
|
cam_tilt = g.cam_tilt_center;
|
|
break;
|
|
case JSButton::button_function_t::k_mount_tilt_up: {
|
|
uint8_t i;
|
|
|
|
// Find the first aux channel configured as mount tilt, if any
|
|
if (SRV_Channels::find_channel(SRV_Channel::k_mount_tilt, i)) {
|
|
|
|
// Get the channel output limits
|
|
SRV_Channel *ch = SRV_Channels::srv_channel(i);
|
|
uint16_t min = ch->get_output_min();
|
|
uint16_t max = ch->get_output_max();
|
|
|
|
cam_tilt = constrain_int16(cam_tilt-g.cam_tilt_step,min,max);
|
|
}
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_mount_tilt_down: {
|
|
uint8_t i;
|
|
|
|
// Find the first aux channel configured as mount tilt, if any
|
|
if (SRV_Channels::find_channel(SRV_Channel::k_mount_tilt, i)) {
|
|
|
|
// Get the channel output limits
|
|
SRV_Channel *ch = SRV_Channels::srv_channel(i);
|
|
uint16_t min = ch->get_output_min();
|
|
uint16_t max = ch->get_output_max();
|
|
|
|
cam_tilt = constrain_int16(cam_tilt+g.cam_tilt_step,min,max);
|
|
}
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_camera_trigger:
|
|
break;
|
|
case JSButton::button_function_t::k_camera_source_toggle:
|
|
if (!held) {
|
|
static bool video_toggle = false;
|
|
video_toggle = !video_toggle;
|
|
if (video_toggle) {
|
|
video_switch = 1900;
|
|
gcs_send_text(MAV_SEVERITY_INFO,"Video Toggle: Source 2");
|
|
} else {
|
|
video_switch = 1100;
|
|
gcs_send_text(MAV_SEVERITY_INFO,"Video Toggle: Source 1");
|
|
}
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_mount_pan_right:
|
|
// Not implemented
|
|
break;
|
|
case JSButton::button_function_t::k_mount_pan_left:
|
|
// Not implemented
|
|
break;
|
|
case JSButton::button_function_t::k_lights1_cycle:
|
|
if (!held) {
|
|
static bool increasing = true;
|
|
if (increasing) {
|
|
lights1 = constrain_float(lights1+g.lights_step,1100,1900);
|
|
} else {
|
|
lights1 = constrain_float(lights1-g.lights_step,1100,1900);
|
|
}
|
|
if (lights1 >= 1900 || lights1 <= 1100) {
|
|
increasing = !increasing;
|
|
}
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_lights1_brighter:
|
|
if (!held) {
|
|
lights1 = constrain_float(lights1+g.lights_step,1100,1900);
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_lights1_dimmer:
|
|
if (!held) {
|
|
lights1 = constrain_float(lights1-g.lights_step,1100,1900);
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_lights2_cycle:
|
|
if (!held) {
|
|
static bool increasing = true;
|
|
if (increasing) {
|
|
lights2 = constrain_float(lights2+g.lights_step,1100,1900);
|
|
} else {
|
|
lights2 = constrain_float(lights2-g.lights_step,1100,1900);
|
|
}
|
|
if (lights2 >= 1900 || lights2 <= 1100) {
|
|
increasing = !increasing;
|
|
}
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_lights2_brighter:
|
|
if (!held) {
|
|
lights2 = constrain_float(lights2+g.lights_step,1100,1900);
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_lights2_dimmer:
|
|
if (!held) {
|
|
lights2 = constrain_float(lights2-g.lights_step,1100,1900);
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_gain_toggle:
|
|
if (!held) {
|
|
static bool lowGain = false;
|
|
lowGain = !lowGain;
|
|
if (lowGain) {
|
|
gain = 0.5f;
|
|
} else {
|
|
gain = 1.0f;
|
|
}
|
|
gcs_send_text_fmt(MAV_SEVERITY_INFO,"#Gain: %2.0f%%",(double)gain*100);
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_gain_inc:
|
|
if (!held) {
|
|
// check that our gain parameters are in correct range, update in eeprom and notify gcs if needed
|
|
g.minGain.set_and_save(constrain_float(g.minGain, 0.10, 0.80));
|
|
g.maxGain.set_and_save(constrain_float(g.maxGain, g.minGain, 1.0));
|
|
g.numGainSettings.set_and_save(constrain_int16(g.numGainSettings, 1, 10));
|
|
|
|
if (g.numGainSettings == 1) {
|
|
gain = constrain_float(g.gain_default, g.minGain, g.maxGain);
|
|
} else {
|
|
gain = constrain_float(gain + (g.maxGain-g.minGain)/(g.numGainSettings-1), g.minGain, g.maxGain);
|
|
}
|
|
|
|
gcs_send_text_fmt(MAV_SEVERITY_INFO,"#Gain is %2.0f%%",(double)gain*100);
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_gain_dec:
|
|
if (!held) {
|
|
// check that our gain parameters are in correct range, update in eeprom and notify gcs if needed
|
|
g.minGain.set_and_save(constrain_float(g.minGain, 0.10, 0.80));
|
|
g.maxGain.set_and_save(constrain_float(g.maxGain, g.minGain, 1.0));
|
|
g.numGainSettings.set_and_save(constrain_int16(g.numGainSettings, 1, 10));
|
|
|
|
if (g.numGainSettings == 1) {
|
|
gain = constrain_float(g.gain_default, g.minGain, g.maxGain);
|
|
} else {
|
|
gain = constrain_float(gain - (g.maxGain-g.minGain)/(g.numGainSettings-1), g.minGain, g.maxGain);
|
|
}
|
|
|
|
gcs_send_text_fmt(MAV_SEVERITY_INFO,"#Gain is %2.0f%%",(double)gain*100);
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_trim_roll_inc:
|
|
rollTrim = constrain_float(rollTrim+10,-200,200);
|
|
break;
|
|
case JSButton::button_function_t::k_trim_roll_dec:
|
|
rollTrim = constrain_float(rollTrim-10,-200,200);
|
|
break;
|
|
case JSButton::button_function_t::k_trim_pitch_inc:
|
|
pitchTrim = constrain_float(pitchTrim+10,-200,200);
|
|
break;
|
|
case JSButton::button_function_t::k_trim_pitch_dec:
|
|
pitchTrim = constrain_float(pitchTrim-10,-200,200);
|
|
break;
|
|
case JSButton::button_function_t::k_input_hold_toggle:
|
|
if (!held) {
|
|
zTrim = z_last-500;
|
|
xTrim = x_last;
|
|
yTrim = y_last;
|
|
gcs_send_text(MAV_SEVERITY_INFO,"#Input Hold Set");
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_relay_1_on:
|
|
relay.on(0);
|
|
break;
|
|
case JSButton::button_function_t::k_relay_1_off:
|
|
relay.off(0);
|
|
break;
|
|
case JSButton::button_function_t::k_relay_1_toggle:
|
|
if (!held) {
|
|
relay.toggle(0);
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_relay_2_on:
|
|
relay.on(1);
|
|
break;
|
|
case JSButton::button_function_t::k_relay_2_off:
|
|
relay.off(1);
|
|
break;
|
|
case JSButton::button_function_t::k_relay_2_toggle:
|
|
if (!held) {
|
|
relay.toggle(1);
|
|
}
|
|
break;
|
|
case JSButton::button_function_t::k_custom_1:
|
|
// Not implemented
|
|
break;
|
|
case JSButton::button_function_t::k_custom_2:
|
|
// Not implemented
|
|
break;
|
|
case JSButton::button_function_t::k_custom_3:
|
|
// Not implemented
|
|
break;
|
|
case JSButton::button_function_t::k_custom_4:
|
|
// Not implemented
|
|
break;
|
|
case JSButton::button_function_t::k_custom_5:
|
|
// Not implemented
|
|
break;
|
|
case JSButton::button_function_t::k_custom_6:
|
|
// Not implemented
|
|
break;
|
|
}
|
|
|
|
// Update the control mode if needed
|
|
if (control_mode != next_mode) {
|
|
if (set_mode(next_mode, MODE_REASON_TX_COMMAND)) {
|
|
// Notify user
|
|
if (ap.initialised) {
|
|
AP_Notify::events.user_mode_change = 1;
|
|
}
|
|
// Update CH5 pwm value (For GCS)
|
|
mode_switch_pwm = next_mode_switch_pwm;
|
|
} else {
|
|
// Notify user
|
|
if (ap.initialised) {
|
|
AP_Notify::events.user_mode_change_failed = 1;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
JSButton* Sub::get_button(uint8_t index)
|
|
{
|
|
// Help to access appropriate parameter
|
|
switch (index) {
|
|
case 0:
|
|
return &g.jbtn_0;
|
|
case 1:
|
|
return &g.jbtn_1;
|
|
case 2:
|
|
return &g.jbtn_2;
|
|
case 3:
|
|
return &g.jbtn_3;
|
|
case 4:
|
|
return &g.jbtn_4;
|
|
case 5:
|
|
return &g.jbtn_5;
|
|
case 6:
|
|
return &g.jbtn_6;
|
|
case 7:
|
|
return &g.jbtn_7;
|
|
case 8:
|
|
return &g.jbtn_8;
|
|
case 9:
|
|
return &g.jbtn_9;
|
|
case 10:
|
|
return &g.jbtn_10;
|
|
case 11:
|
|
return &g.jbtn_11;
|
|
case 12:
|
|
return &g.jbtn_12;
|
|
case 13:
|
|
return &g.jbtn_13;
|
|
case 14:
|
|
return &g.jbtn_14;
|
|
case 15:
|
|
return &g.jbtn_15;
|
|
default:
|
|
return &g.jbtn_0;
|
|
}
|
|
}
|
|
|
|
void Sub::default_js_buttons()
|
|
{
|
|
JSButton::button_function_t defaults[16][2] = {
|
|
{JSButton::button_function_t::k_none, JSButton::button_function_t::k_none},
|
|
{JSButton::button_function_t::k_mode_1, JSButton::button_function_t::k_none},
|
|
{JSButton::button_function_t::k_mode_3, JSButton::button_function_t::k_none},
|
|
{JSButton::button_function_t::k_mode_2, JSButton::button_function_t::k_none},
|
|
|
|
{JSButton::button_function_t::k_disarm, JSButton::button_function_t::k_none},
|
|
{JSButton::button_function_t::k_shift, JSButton::button_function_t::k_none},
|
|
{JSButton::button_function_t::k_arm, JSButton::button_function_t::k_none},
|
|
{JSButton::button_function_t::k_mount_center, JSButton::button_function_t::k_none},
|
|
|
|
{JSButton::button_function_t::k_input_hold_toggle, JSButton::button_function_t::k_none},
|
|
{JSButton::button_function_t::k_mount_tilt_down, JSButton::button_function_t::k_none},
|
|
{JSButton::button_function_t::k_mount_tilt_up, JSButton::button_function_t::k_none},
|
|
{JSButton::button_function_t::k_gain_inc, JSButton::button_function_t::k_trim_pitch_dec},
|
|
|
|
{JSButton::button_function_t::k_gain_dec, JSButton::button_function_t::k_trim_pitch_inc},
|
|
{JSButton::button_function_t::k_lights1_dimmer, JSButton::button_function_t::k_trim_roll_dec},
|
|
{JSButton::button_function_t::k_lights1_brighter, JSButton::button_function_t::k_trim_roll_inc},
|
|
{JSButton::button_function_t::k_none, JSButton::button_function_t::k_none},
|
|
};
|
|
|
|
for (int i = 0; i < 16; i++) {
|
|
get_button(i)->set_default(defaults[i][0], defaults[i][1]);
|
|
}
|
|
}
|
|
|
|
void Sub::set_neutral_controls()
|
|
{
|
|
int16_t channels[11];
|
|
|
|
for (uint8_t i = 0; i < 7; i++) {
|
|
channels[i] = 1500;
|
|
}
|
|
|
|
for (uint8_t i = 7; i < 11; i++) {
|
|
channels[i] = 0xffff;
|
|
}
|
|
|
|
channels[4] = 0xffff; // Leave mode switch where it was
|
|
|
|
failsafe.rc_override_active = hal.rcin->set_overrides(channels, 10);
|
|
}
|