00001 #include "teleop_twist.h"
00002 #include "ardrone_autonomy/LedAnim.h"
00003 #include "utils/ardrone_date.h"
00004 
00005 inline float max(float a, float b) { return a > b ? a : b; }
00006 inline float min(float a, float b) { return a < b ? a : b; }
00007 
00008 bool needs_takeoff = false;
00009 bool needs_land = false;
00010 bool needs_reset = false;
00011 geometry_msgs::Twist cmd_vel;
00012 float old_left_right = -10.0;
00013 float old_front_back = -10.0;
00014 float old_up_down = -10.0;
00015 float old_turn = -10.0;
00016 
00017 int cam_state = DEFAULT_CAM_STATE; 
00018 int set_navdata_demo_value = DEFAULT_NAVDATA_DEMO; 
00019 int32_t detect_enemy_color = ARDRONE_DETECTION_COLOR_ORANGE_YELLOW;
00020 int32_t detect_dtype = CAD_TYPE_MULTIPLE_DETECTION_MODE;
00021 int32_t detect_hori_type = TAG_TYPE_MASK(TAG_TYPE_SHELL_TAG_V2);
00022 int32_t detect_vert_type = TAG_TYPE_MASK(TAG_TYPE_BLACK_ROUNDEL);
00023 int32_t detect_indoor_hull = 0;
00024 int32_t detect_disable_placeholder = 0;
00025 int32_t detect_enable_placeholder = 1;
00026 
00027 const LED_ANIMATION_IDS ledAnimMap[14] = {
00028         BLINK_GREEN_RED, BLINK_GREEN, BLINK_RED, BLINK_ORANGE,
00029         SNAKE_GREEN_RED, FIRE, STANDARD, RED, GREEN, RED_SNAKE,BLANK,
00030         LEFT_GREEN_RIGHT_RED, LEFT_RED_RIGHT_GREEN, BLINK_STANDARD};
00031 
00032 
00033 bool setCamChannelCallback(ardrone_autonomy::CamSelect::Request& request, ardrone_autonomy::CamSelect::Response& response)
00034 {
00035     const int _modes = (IS_ARDRONE1) ? 4 : 2;
00036     cam_state = request.channel % _modes;
00037     ARDRONE_TOOL_CONFIGURATION_ADDEVENT (video_channel, &cam_state, NULL);
00038     fprintf(stderr, "\nSetting camera channel to : %d.\n", cam_state);
00039     response.result = true;
00040     return true;
00041 }
00042 
00043 bool toggleCamCallback(std_srvs::Empty::Request& request, std_srvs::Empty::Response& response)
00044 {
00045     const int _modes = (IS_ARDRONE1) ? 4 : 2;
00046     cam_state = (cam_state + 1) % _modes;
00047     ARDRONE_TOOL_CONFIGURATION_ADDEVENT (video_channel, &cam_state, NULL);
00048     fprintf(stderr, "\nSetting camera channel to : %d.\n", cam_state);
00049     return true;
00050 }
00051 
00052 
00053 bool setRecordCallback(ardrone_autonomy::RecordEnable::Request &request, ardrone_autonomy::RecordEnable::Response& response)
00054 {
00055   char record_command[ARDRONE_DATE_MAXSIZE + 64];
00056   int32_t new_codec;
00057 
00058   if( request.enable == true ) {
00059     char date[ARDRONE_DATE_MAXSIZE];
00060     time_t t = time(NULL);
00061     
00062     
00063     strftime(date, ARDRONE_DATE_MAXSIZE, ARDRONE_FILE_DATE_FORMAT, localtime(&t));
00064     snprintf(record_command, sizeof(record_command), "%d,%s", USERBOX_CMD_START, date);
00065     new_codec = MP4_360P_H264_720P_CODEC;
00066   } else {
00067     snprintf(record_command, sizeof(record_command), "%d", USERBOX_CMD_STOP );
00068     new_codec = H264_360P_CODEC;
00069   }
00070 
00071   vp_os_mutex_lock(&twist_lock);
00072   ARDRONE_TOOL_CONFIGURATION_ADDEVENT (video_codec, &new_codec, NULL );
00073   ARDRONE_TOOL_CONFIGURATION_ADDEVENT (userbox_cmd, record_command, NULL );
00074   vp_os_mutex_unlock(&twist_lock);
00075 
00076   response.result = true;
00077   return true;
00078 }    
00079 
00080 bool setLedAnimationCallback(ardrone_autonomy::LedAnim::Request& request, ardrone_autonomy::LedAnim::Response& response)
00081 {
00082     LED_ANIMATION_IDS anim_id = ledAnimMap[request.type % 14]; 
00083     vp_os_mutex_lock(&twist_lock);
00084     ardrone_at_set_led_animation(anim_id, (float) fabs(request.freq), (uint32_t) abs(request.duration));
00085     vp_os_mutex_unlock(&twist_lock);
00086     response.result = true;
00087     return true;
00088 }
00089 
00090 bool setFlightAnimationCallback(ardrone_autonomy::FlightAnim::Request &request, ardrone_autonomy::FlightAnim::Response &response)
00091 {
00092     char param[20];
00093     const int anim_type = request.type % ARDRONE_NB_ANIM_MAYDAY;
00094     const int anim_duration = (request.duration > 0) ? request.duration : MAYDAY_TIMEOUT[anim_type];
00095     snprintf(param, sizeof (param), "%d,%d", anim_type, anim_duration);
00096     vp_os_mutex_lock(&twist_lock);
00097     ARDRONE_TOOL_CONFIGURATION_ADDEVENT(flight_anim, param, NULL);
00098     vp_os_mutex_unlock(&twist_lock);
00099     response.result = true;
00100     return true;
00101 }
00102 
00103 bool flatTrimCallback(std_srvs::Empty::Request &request, std_srvs::Empty::Response &response)
00104 {
00105     vp_os_mutex_lock(&twist_lock);
00106     ardrone_at_set_flat_trim();
00107     vp_os_mutex_unlock(&twist_lock);
00108     fprintf(stderr, "\nFlat Trim Set.\n");
00109 }
00110 
00111 void cmdVelCallback(const geometry_msgs::TwistConstPtr &msg)
00112 {
00113     vp_os_mutex_lock(&twist_lock);
00114     
00115     cmd_vel.linear.x  = max(min(-msg->linear.x, 1.0), -1.0);
00116     cmd_vel.linear.y  = max(min(-msg->linear.y, 1.0), -1.0);
00117         cmd_vel.linear.z  = max(min(msg->linear.z, 1.0), -1.0);
00118         cmd_vel.angular.z = max(min(-msg->angular.z, 1.0), -1.0);
00119     
00120     
00121     cmd_vel.angular.x = msg->angular.x;
00122     cmd_vel.angular.y = msg->angular.y;
00123     vp_os_mutex_unlock(&twist_lock);
00124 }
00125 
00126 void landCallback(const std_msgs::Empty &msg)
00127 {
00128     vp_os_mutex_lock(&twist_lock);
00129     needs_land = true;
00130     vp_os_mutex_unlock(&twist_lock);
00131 }
00132 
00133 void resetCallback(const std_msgs::Empty &msg)
00134 {       
00135     vp_os_mutex_lock(&twist_lock);
00136     needs_reset = true;
00137     vp_os_mutex_unlock(&twist_lock);
00138 }
00139 
00140 void takeoffCallback(const std_msgs::Empty &msg)
00141 {
00142     vp_os_mutex_lock(&twist_lock);
00143     needs_takeoff = true;
00144     vp_os_mutex_unlock(&twist_lock);
00145 }
00146 
00147 C_RESULT open_teleop(void)
00148 {
00149         return C_OK;
00150 }
00151 
00152 C_RESULT update_teleop(void)
00153 {
00154         
00155         
00156     vp_os_mutex_lock(&twist_lock);
00157     if (needs_reset)
00158     {
00159         ardrone_tool_set_ui_pad_select(1);
00160         needs_reset = false;
00161     }
00162     else if (needs_takeoff)
00163     {
00164         ardrone_tool_set_ui_pad_start(1);
00165         needs_takeoff = false;
00166     }
00167     else if (needs_land)
00168     {
00169         ardrone_tool_set_ui_pad_start(0);
00170         needs_land = false;
00171     }
00172     else
00173     {
00174 
00175         float left_right = (float) cmd_vel.linear.y;
00176         float front_back = (float) cmd_vel.linear.x;
00177         float up_down = (float) cmd_vel.linear.z;
00178         float turn = (float) cmd_vel.angular.z;
00179         
00180         bool is_changed = !(
00181                 (fabs(left_right - old_left_right) < _EPS) && 
00182                 (fabs(front_back - old_front_back) < _EPS) && 
00183                 (fabs(up_down - old_up_down) < _EPS) && 
00184                 (fabs(turn - old_turn) < _EPS)
00185                 );
00186         
00187         
00188         
00189         
00190         
00191         int32_t control_flag = 0x00;
00192         int32_t combined_yaw = 0x00;
00193         
00194         
00195         int32_t hover = (int32_t)
00196                 (
00197                 (fabs(left_right) < _EPS) && 
00198                 (fabs(front_back) < _EPS) && 
00199                 (fabs(up_down) < _EPS) && 
00200                 (fabs(turn) < _EPS) &&
00201                 
00202                 
00203                 (fabs(cmd_vel.angular.x) < _EPS) &&
00204                 (fabs(cmd_vel.angular.y) < _EPS)
00205                 );
00206 
00207         control_flag |= ((1 - hover) << 0);
00208         control_flag |= (combined_yaw << 1);
00209         
00210         
00211         old_left_right = left_right;
00212         old_front_back = front_back;
00213         old_up_down = up_down;
00214         old_turn = turn;
00215         
00216         if ((is_changed) || (hover))
00217         {
00218             ardrone_tool_set_progressive_cmd(control_flag, left_right, front_back, up_down, turn, 0.0, 0.0);
00219         }
00220 
00221     }
00222     vp_os_mutex_unlock(&twist_lock);
00223         return C_OK;
00224 }
00225 
00226 C_RESULT close_teleop(void)
00227 {
00228         return C_OK;
00229 }
00230 
00231 input_device_t teleop = {
00232         "Teleop",
00233         open_teleop,
00234         update_teleop,
00235         close_teleop
00236 };
00237