2021-02-14 12:50:42 +08:00
|
|
|
#include "moveTaiChi.h" //轮胎运动库
|
|
|
|
|
#include "sensorTaiChi.h" //传感器库
|
|
|
|
|
#include "servoTaiChi.h" //舵机库
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
Move move; //轮胎运动实例
|
|
|
|
|
Sensor sensor; //传感器实例
|
|
|
|
|
Servo servo; //舵机实例
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//****************************************运动路径****************************************
|
|
|
|
|
//坐标点操作定义
|
2021-02-17 19:52:59 +08:00
|
|
|
#define NORMAL_POINT 0 //普通点
|
|
|
|
|
#define CATCH_POINT 1 //抓取点
|
|
|
|
|
#define RELEASE_POINT 2 //释放点(使用机械臂)
|
2021-02-21 19:28:17 +08:00
|
|
|
#define CARRY_POINT 3 //携带点(从底盘)
|
2021-02-17 19:52:59 +08:00
|
|
|
#define GETOUT_POINT 4 //释放点(从底盘)
|
2021-02-14 12:50:42 +08:00
|
|
|
|
|
|
|
|
//坐标点数组定义
|
|
|
|
|
#define X 0
|
|
|
|
|
#define Y 1
|
|
|
|
|
#define TYPE 2
|
|
|
|
|
|
|
|
|
|
int8_t route[][3] =
|
|
|
|
|
{
|
|
|
|
|
{0, 0, NORMAL_POINT},
|
|
|
|
|
{0, 1, NORMAL_POINT},
|
2021-02-18 15:13:40 +08:00
|
|
|
{1, 1, CARRY_POINT},
|
|
|
|
|
{2, 1, NORMAL_POINT},
|
|
|
|
|
{2, 2, CATCH_POINT},
|
|
|
|
|
{2, 3, NORMAL_POINT},
|
|
|
|
|
{3, 3, GETOUT_POINT},
|
2021-02-14 23:38:28 +08:00
|
|
|
{2, 3, NORMAL_POINT},
|
|
|
|
|
{2, 4, NORMAL_POINT},
|
|
|
|
|
{3, 4, RELEASE_POINT},
|
|
|
|
|
{2, 4, NORMAL_POINT},
|
|
|
|
|
{2, 3, NORMAL_POINT},
|
2021-02-18 15:13:40 +08:00
|
|
|
{1, 3, NORMAL_POINT},
|
|
|
|
|
{0, 3, NORMAL_POINT},
|
|
|
|
|
{0, 2, CATCH_POINT},
|
|
|
|
|
{1, 2, NORMAL_POINT},
|
|
|
|
|
{2, 2, NORMAL_POINT},
|
2021-02-14 23:38:28 +08:00
|
|
|
{3, 2, RELEASE_POINT},
|
2021-02-14 12:50:42 +08:00
|
|
|
{2, 2, NORMAL_POINT},
|
2021-02-18 15:13:40 +08:00
|
|
|
{2, 3, NORMAL_POINT},
|
|
|
|
|
{3, 3, GETOUT_POINT},
|
|
|
|
|
{2, 3, NORMAL_POINT}
|
2021-02-14 12:50:42 +08:00
|
|
|
};
|
|
|
|
|
//***************************************************************************************
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//****************************************可调参数****************************************
|
2021-02-18 11:31:28 +08:00
|
|
|
//抓取点移动用时
|
|
|
|
|
#define CATCH_MOVE_DELAY_TIME 2000
|
|
|
|
|
//释放点移动用时
|
|
|
|
|
#define RELEASE_MOVE_DELAY_TIME 2000
|
2021-02-14 15:44:39 +08:00
|
|
|
|
2021-02-14 12:50:42 +08:00
|
|
|
//重置留时
|
2021-02-18 11:32:27 +08:00
|
|
|
#define RESET_DELAY_TIME 5000
|
2021-02-14 12:50:42 +08:00
|
|
|
//抓取留时
|
2021-02-18 11:32:27 +08:00
|
|
|
#define CATCH_DELAY_TIME 5000
|
2021-02-14 12:50:42 +08:00
|
|
|
//释放留时
|
2021-02-18 11:32:27 +08:00
|
|
|
#define RELEASE_DELAY_TIME 5000
|
2021-02-14 12:50:42 +08:00
|
|
|
|
2021-02-18 13:19:34 +08:00
|
|
|
//最大抓取尝试次数
|
|
|
|
|
#define MAX_CATCH_TIMES 2
|
|
|
|
|
|
2021-02-14 12:50:42 +08:00
|
|
|
//传感器判断转向完成延时
|
|
|
|
|
#define TRUN_CHECK_DELAY 1000
|
|
|
|
|
|
|
|
|
|
//修正循迹时单侧减速比率
|
|
|
|
|
#define LINE_FIX_SPEED_RATE 0.8
|
2021-02-15 13:24:30 +08:00
|
|
|
//低速重新寻线时减速比率
|
|
|
|
|
#define LINE_FIND_SPEED_RATE 0.5
|
2021-02-14 12:50:42 +08:00
|
|
|
//***************************************************************************************
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//****************************************全局变量****************************************
|
2021-02-14 23:38:28 +08:00
|
|
|
//数组位置标记
|
2021-02-14 12:50:42 +08:00
|
|
|
int passed_flag = 0;
|
2021-02-14 23:38:28 +08:00
|
|
|
int next_flag = 1;
|
|
|
|
|
int next_next_flag = 2;
|
|
|
|
|
|
|
|
|
|
//最大数组位置标记
|
|
|
|
|
int max_flag;
|
2021-02-14 12:50:42 +08:00
|
|
|
|
|
|
|
|
#define FRONT_NEXT 0
|
|
|
|
|
#define BACK_NEXT 1
|
|
|
|
|
//下一点朝向
|
|
|
|
|
uint8_t next_position = FRONT_NEXT;
|
2021-02-18 13:19:34 +08:00
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
//爪子是否抓有环
|
|
|
|
|
bool is_claw_catch = false;
|
|
|
|
|
|
2021-02-18 13:19:34 +08:00
|
|
|
//爪子是否正常
|
|
|
|
|
bool is_claw_ok = true;
|
2021-02-21 19:28:17 +08:00
|
|
|
|
|
|
|
|
//底盘是否携带环
|
|
|
|
|
bool is_carry = false;
|
2021-02-14 12:50:42 +08:00
|
|
|
//***************************************************************************************
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//****************************************自定函数****************************************
|
2021-02-21 19:28:17 +08:00
|
|
|
#define NO_JUMP 0
|
|
|
|
|
#define JUMP_DEAD_ROAD 1
|
|
|
|
|
//更新标记
|
|
|
|
|
void UpdateFlag(uint8_t jump_choice = NO_JUMP);
|
|
|
|
|
|
2021-02-15 10:58:46 +08:00
|
|
|
//计算方向,同时更改完成转向后相对下一点的朝向
|
2021-02-14 12:50:42 +08:00
|
|
|
uint8_t CalcDirection(void);
|
|
|
|
|
|
|
|
|
|
#define FRONT_END 0
|
|
|
|
|
#define BACK_END 1
|
|
|
|
|
#define CATCH_END 2
|
|
|
|
|
#define RELEASE_END 3
|
2021-02-18 14:31:51 +08:00
|
|
|
//沿线直行,在触发条件后离开函数但不停止
|
2021-02-15 11:01:17 +08:00
|
|
|
void LineForward(uint8_t end_position, float speed_rate = 1.0);
|
2021-02-14 12:50:42 +08:00
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
//沿线后退,在触发条件后离开函数但不停止
|
2021-02-15 11:01:17 +08:00
|
|
|
void LineBackward(uint8_t end_position, float speed_rate = 1.0);
|
2021-02-14 12:50:42 +08:00
|
|
|
|
2021-02-21 19:28:17 +08:00
|
|
|
//直行或后退或转向,完成后离开函数但不停止。会自动跳过无需前往的释放点
|
2021-02-15 19:17:59 +08:00
|
|
|
void TurnDirection(float speed_rate = 1.0);
|
2021-02-17 19:35:43 +08:00
|
|
|
|
|
|
|
|
//抓取环,并判定是否抓取成功
|
2021-02-18 14:31:51 +08:00
|
|
|
bool CatchAndCheck(float speed = 1.0);
|
2021-02-18 13:33:55 +08:00
|
|
|
|
|
|
|
|
//打开爪子,并判断爪子是否能正常工作
|
|
|
|
|
bool OpenClawAndCheck(void);
|
2021-02-14 12:50:42 +08:00
|
|
|
//***************************************************************************************
|
2021-02-10 20:13:28 +08:00
|
|
|
|
|
|
|
|
|
2021-02-18 15:13:40 +08:00
|
|
|
//****************************************调试相关****************************************
|
|
|
|
|
//注释以关闭调试功能
|
|
|
|
|
#define TAICHI_DEBUG
|
|
|
|
|
|
|
|
|
|
#ifdef TAICHI_DEBUG
|
|
|
|
|
|
|
|
|
|
#define DEBUG_BAUT_RATE 115200
|
|
|
|
|
#define DEBUG_PAUSE_INTERRUPTNUM 2 //PIN 21
|
|
|
|
|
#define DEBUG_PAUSE_PIN 21
|
|
|
|
|
|
|
|
|
|
int loop_time = 0;
|
|
|
|
|
|
|
|
|
|
//中断函数,用于调试时暂停程序
|
|
|
|
|
void DebugPause(void)
|
|
|
|
|
{
|
|
|
|
|
while (digitalRead(DEBUG_PAUSE_PIN) == LOW) {}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//错误消息函数,用于在出现致命错误后结束程序
|
|
|
|
|
void DebugCanNotContinue(char* message)
|
|
|
|
|
{
|
|
|
|
|
Serial.print("#TAICHI: CAN NOT CONTINUE WHEN "); Serial.println(message);
|
|
|
|
|
Serial.print("#TAICHI: loop_time: "); Serial.println(loop_time);
|
|
|
|
|
Serial.print("#TAICHI: pass: ["); Serial.print(route[passed_flag][X]); Serial.print(", "); Serial.print(route[passed_flag][Y]); Serial.print("]");
|
|
|
|
|
Serial.print(" flag: "); Serial.print(passed_flag);
|
|
|
|
|
Serial.print(" TYPE: "); Serial.println((int)route[passed_flag][TYPE]);
|
|
|
|
|
Serial.print("#TAICHI: next: ["); Serial.print(route[next_flag][X]); Serial.print(", "); Serial.print(route[next_flag][Y]); Serial.print("]");
|
|
|
|
|
Serial.print(" next_position: "); Serial.print((int)next_position);
|
|
|
|
|
Serial.print(" TYPE: "); Serial.println((int)route[next_flag][TYPE]);
|
|
|
|
|
Serial.print("#TAICHI: is_claw_catch: "); Serial.print((int)is_claw_catch); Serial.print(" is_claw_ok: "); Serial.println((int)is_claw_ok);
|
|
|
|
|
|
|
|
|
|
while (1) {}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
//***************************************************************************************
|
|
|
|
|
|
|
|
|
|
|
2021-02-17 12:42:28 +08:00
|
|
|
void setup()
|
2021-02-10 20:13:28 +08:00
|
|
|
{
|
2021-02-15 12:28:14 +08:00
|
|
|
#ifdef TAICHI_DEBUG
|
2021-02-17 12:42:28 +08:00
|
|
|
attachInterrupt(DEBUG_PAUSE_INTERRUPTNUM, DebugPause, LOW);
|
2021-02-17 13:21:28 +08:00
|
|
|
Serial.begin(DEBUG_BAUT_RATE);
|
2021-02-15 13:14:46 +08:00
|
|
|
Serial.println("#TAICHI: ======================setup()=====================");
|
2021-02-15 12:28:14 +08:00
|
|
|
#endif
|
|
|
|
|
|
2021-02-14 12:50:42 +08:00
|
|
|
move.Stop();
|
|
|
|
|
servo.StopAndReset();
|
|
|
|
|
|
2021-02-14 23:38:28 +08:00
|
|
|
//计算最大数组下标
|
|
|
|
|
max_flag = sizeof(route) / sizeof(route[0]) - 1;
|
|
|
|
|
|
2021-02-14 12:50:42 +08:00
|
|
|
//前往 0, 0
|
2021-02-18 14:31:51 +08:00
|
|
|
//沿线直行,到后端传感器接触下一条线离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
LineForward(BACK_END);
|
2021-02-10 20:13:28 +08:00
|
|
|
|
2021-02-14 12:50:42 +08:00
|
|
|
//已越过 0, 0 正式进入循环
|
2021-02-10 20:13:28 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-02-14 12:50:42 +08:00
|
|
|
void loop()
|
2021-02-10 20:13:28 +08:00
|
|
|
{
|
2021-02-15 13:14:46 +08:00
|
|
|
#ifdef TAICHI_DEBUG
|
2021-02-17 12:42:28 +08:00
|
|
|
loop_time++;
|
2021-02-15 13:14:46 +08:00
|
|
|
Serial.println("#TAICHI: ====================New loop()====================");
|
2021-02-17 12:42:28 +08:00
|
|
|
Serial.print("#TAICHI: loop_time: "); Serial.println(loop_time);
|
2021-02-18 11:24:54 +08:00
|
|
|
Serial.print("#TAICHI: pass: ["); Serial.print(route[passed_flag][X]); Serial.print(", "); Serial.print(route[passed_flag][Y]); Serial.print("]");
|
|
|
|
|
Serial.print(" flag: "); Serial.print(passed_flag);
|
|
|
|
|
Serial.print(" TYPE: "); Serial.println((int)route[passed_flag][TYPE]);
|
|
|
|
|
Serial.print("#TAICHI: next: ["); Serial.print(route[next_flag][X]); Serial.print(", "); Serial.print(route[next_flag][Y]); Serial.print("]");
|
|
|
|
|
Serial.print(" next_position: "); Serial.print((int)next_position);
|
|
|
|
|
Serial.print(" TYPE: "); Serial.println((int)route[next_flag][TYPE]);
|
2021-02-18 14:31:51 +08:00
|
|
|
Serial.print("#TAICHI: is_claw_catch: "); Serial.print((int)is_claw_catch); Serial.print(" is_claw_ok: "); Serial.println((int)is_claw_ok);
|
2021-02-15 13:14:46 +08:00
|
|
|
#endif
|
|
|
|
|
|
2021-02-21 19:28:17 +08:00
|
|
|
int8_t& passed_flag_type = route[passed_flag][TYPE];
|
|
|
|
|
int8_t& next_flag_type = route[next_flag][TYPE];
|
|
|
|
|
int8_t& next_next_flag_type = route[next_next_flag][TYPE];
|
|
|
|
|
|
|
|
|
|
//情况一:刚完整经过普通点,下一个点为普通点或携带点
|
|
|
|
|
if (passed_flag_type == NORMAL_POINT && (next_flag_type == NORMAL_POINT || next_flag_type == CARRY_POINT))
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
2021-02-15 10:58:46 +08:00
|
|
|
if (next_position == FRONT_NEXT)
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
2021-02-18 14:31:51 +08:00
|
|
|
//沿线直行,到前端传感器接触下一条线离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
LineForward(FRONT_END);
|
2021-02-21 19:28:17 +08:00
|
|
|
|
|
|
|
|
//若下一点为携带点
|
|
|
|
|
if (next_flag_type == CARRY_POINT)
|
|
|
|
|
{
|
|
|
|
|
//底盘携带
|
|
|
|
|
is_carry = true;
|
|
|
|
|
|
|
|
|
|
//越过点成为普通点
|
|
|
|
|
next_flag_type = NORMAL_POINT;
|
|
|
|
|
}
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2021-02-18 14:31:51 +08:00
|
|
|
//沿线后退,到后端传感器接触下一条线离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
LineBackward(BACK_END);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//继续直行或后退或转向
|
2021-02-15 19:17:59 +08:00
|
|
|
TurnDirection();
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
//情况二:刚完整经过普通点,下一个点为抓取点
|
2021-02-21 19:28:17 +08:00
|
|
|
else if (passed_flag_type == NORMAL_POINT && next_flag_type == CATCH_POINT)
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
2021-02-18 14:31:51 +08:00
|
|
|
if (!is_claw_catch && !is_claw_ok) //爪子上无环且状态不正常
|
|
|
|
|
{
|
|
|
|
|
move.Stop(); //停止前进
|
|
|
|
|
OpenClawAndCheck(); //再次检测爪子是否正常
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!is_claw_catch && is_claw_ok) //爪子上无环且状态正常
|
|
|
|
|
{
|
|
|
|
|
//沿线直行,在抓取位置离开函数
|
|
|
|
|
LineForward(CATCH_END);
|
2021-02-14 12:50:42 +08:00
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
//停止前进
|
|
|
|
|
move.Stop();
|
2021-02-14 12:50:42 +08:00
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
//抓取
|
2021-02-21 19:28:17 +08:00
|
|
|
if (!CatchAndCheck())
|
|
|
|
|
is_carry = true; //抓取失败
|
|
|
|
|
}
|
|
|
|
|
else //未进行抓取
|
|
|
|
|
{
|
|
|
|
|
is_carry = true;
|
2021-02-18 14:31:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//继续沿线直行,到前端传感器接触下一条线离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
LineForward(FRONT_END);
|
|
|
|
|
|
2021-02-14 13:44:18 +08:00
|
|
|
//继续直行或后退或转向
|
2021-02-15 19:17:59 +08:00
|
|
|
TurnDirection();
|
2021-02-14 12:50:42 +08:00
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
//将越过的点视为普通点。因为即使抓取失败,环也会被携带在底盘内
|
2021-02-21 19:28:17 +08:00
|
|
|
next_flag_type = NORMAL_POINT;
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
2021-02-17 19:52:59 +08:00
|
|
|
//情况三:刚完整经过普通点,下一个点为释放点(使用机械臂)
|
2021-02-21 19:28:17 +08:00
|
|
|
else if (passed_flag_type == NORMAL_POINT && next_flag_type == RELEASE_POINT)
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
2021-02-18 14:31:51 +08:00
|
|
|
//沿线直行,在释放位置离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
LineForward(RELEASE_END);
|
|
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
//停止前进
|
|
|
|
|
move.Stop();
|
|
|
|
|
|
2021-02-21 19:28:17 +08:00
|
|
|
//释放
|
|
|
|
|
servo.Release();
|
|
|
|
|
delay(RELEASE_DELAY_TIME); //释放留时
|
|
|
|
|
is_claw_catch = false;
|
2021-02-14 12:50:42 +08:00
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
//下一点朝向为后
|
2021-02-14 12:50:42 +08:00
|
|
|
next_position = BACK_NEXT;
|
|
|
|
|
}
|
2021-02-17 19:52:59 +08:00
|
|
|
//情况四:刚完整经过释放点(使用机械臂),下一个点为普通点
|
2021-02-21 19:28:17 +08:00
|
|
|
else if (passed_flag_type == RELEASE_POINT && next_flag_type == NORMAL_POINT)
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
2021-02-18 14:31:51 +08:00
|
|
|
//沿线后退,到后端传感器接触下一条线离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
LineBackward(BACK_END);
|
|
|
|
|
|
2021-02-18 15:13:40 +08:00
|
|
|
//停止后退
|
2021-02-18 14:31:51 +08:00
|
|
|
move.Stop();
|
|
|
|
|
|
2021-02-14 12:50:42 +08:00
|
|
|
//机械臂复原
|
|
|
|
|
servo.Reset();
|
|
|
|
|
delay(RESET_DELAY_TIME); //复原留时
|
|
|
|
|
|
|
|
|
|
//继续后退或转向
|
2021-02-15 19:17:59 +08:00
|
|
|
TurnDirection();
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
2021-02-17 19:52:59 +08:00
|
|
|
//情况五:刚完整经过普通点,下一个点为释放点(从底盘)
|
2021-02-21 19:28:17 +08:00
|
|
|
else if (passed_flag_type == NORMAL_POINT && next_flag_type == GETOUT_POINT)
|
2021-02-17 19:52:59 +08:00
|
|
|
{
|
2021-02-18 14:31:51 +08:00
|
|
|
//沿线直行,到后端传感器接触下一条线离开函数
|
2021-02-17 19:52:59 +08:00
|
|
|
LineForward(BACK_END);
|
|
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
//停止前进
|
|
|
|
|
move.Stop();
|
|
|
|
|
|
2021-02-17 19:52:59 +08:00
|
|
|
//下一点朝向为后
|
|
|
|
|
next_position = BACK_NEXT;
|
|
|
|
|
}
|
|
|
|
|
//情况六:刚完整经过释放点(从底盘),下一个点为普通点
|
2021-02-21 19:28:17 +08:00
|
|
|
else if (passed_flag_type == GETOUT_POINT && next_flag_type == NORMAL_POINT)
|
2021-02-17 19:52:59 +08:00
|
|
|
{
|
2021-02-18 15:13:40 +08:00
|
|
|
//沿线后退,到前端传感器接触线离开函数
|
|
|
|
|
LineBackward(FRONT_END);
|
2021-02-21 19:28:17 +08:00
|
|
|
|
|
|
|
|
is_carry = false;
|
2021-02-18 15:13:40 +08:00
|
|
|
|
|
|
|
|
//沿线后退,到后端传感器接触线离开函数
|
2021-02-17 19:52:59 +08:00
|
|
|
LineBackward(BACK_END);
|
|
|
|
|
|
|
|
|
|
//继续后退或转向
|
|
|
|
|
TurnDirection();
|
|
|
|
|
}
|
2021-02-15 18:28:33 +08:00
|
|
|
//出现错误
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
move.Stop();
|
|
|
|
|
|
|
|
|
|
#ifdef TAICHI_DEBUG
|
2021-02-18 15:13:40 +08:00
|
|
|
DebugCanNotContinue("CHOOSE LOOP");
|
2021-02-15 18:28:33 +08:00
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
while (1) {}
|
|
|
|
|
}
|
2021-02-14 23:38:28 +08:00
|
|
|
|
|
|
|
|
//更新标记,继续循环
|
2021-02-21 19:28:17 +08:00
|
|
|
UpdateFlag();
|
2021-02-15 13:14:46 +08:00
|
|
|
|
|
|
|
|
#ifdef TAICHI_DEBUG
|
|
|
|
|
Serial.println("#TAICHI: ====================End loop()====================");
|
|
|
|
|
#endif
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-02-21 19:28:17 +08:00
|
|
|
//更新标记
|
|
|
|
|
void UpdateFlag(uint8_t jump_choice)
|
|
|
|
|
{
|
|
|
|
|
if (jump_choice == NO_JUMP)
|
|
|
|
|
{
|
|
|
|
|
passed_flag = next_flag;
|
|
|
|
|
|
|
|
|
|
next_flag = next_next_flag;
|
|
|
|
|
|
|
|
|
|
if (++next_next_flag > max_flag)
|
|
|
|
|
next_next_flag = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
int current_passed_flag = passed_flag;
|
|
|
|
|
|
|
|
|
|
UpdateFlag(NO_JUMP);
|
|
|
|
|
UpdateFlag(NO_JUMP);
|
|
|
|
|
|
|
|
|
|
passed_flag = current_passed_flag;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-02-15 10:58:46 +08:00
|
|
|
//计算方向,同时更改完成转向后相对下一点的朝向
|
2021-02-14 12:50:42 +08:00
|
|
|
uint8_t CalcDirection(void)
|
|
|
|
|
{
|
|
|
|
|
//计算第三点与第一点的相对坐标 rx0, ry0
|
2021-02-14 23:38:28 +08:00
|
|
|
int8_t rx0 = route[next_next_flag][X] - route[passed_flag][X];
|
|
|
|
|
int8_t ry0 = route[next_next_flag][Y] - route[passed_flag][Y];
|
2021-02-14 12:50:42 +08:00
|
|
|
|
|
|
|
|
//计算当前小车朝向的方向向量 vx, vy
|
2021-02-14 23:38:28 +08:00
|
|
|
int8_t vx = route[next_flag][X] - route[passed_flag][X];
|
|
|
|
|
int8_t vy = route[next_flag][Y] - route[passed_flag][Y];
|
2021-02-14 12:50:42 +08:00
|
|
|
|
|
|
|
|
//坐标旋转变换
|
|
|
|
|
int8_t rx, ry;
|
|
|
|
|
if (vx == 0 && vy == 1)
|
|
|
|
|
{
|
|
|
|
|
rx = rx0;
|
|
|
|
|
ry = ry0;
|
|
|
|
|
}
|
|
|
|
|
else if (vx == -1 && vy == 0)
|
|
|
|
|
{
|
|
|
|
|
rx = ry0;
|
|
|
|
|
ry = -rx0;
|
|
|
|
|
}
|
|
|
|
|
else if (vx == 0 && vy == -1)
|
|
|
|
|
{
|
|
|
|
|
rx = -rx0;
|
|
|
|
|
ry = -ry0;
|
|
|
|
|
}
|
|
|
|
|
else if (vx == 1 && vy == 0)
|
|
|
|
|
{
|
|
|
|
|
rx = -ry0;
|
|
|
|
|
ry = rx0;
|
|
|
|
|
}
|
2021-02-18 15:13:40 +08:00
|
|
|
else DebugCanNotContinue("CALC DIRECTION"); //调试用
|
2021-02-14 12:50:42 +08:00
|
|
|
|
|
|
|
|
//判断行进方向
|
|
|
|
|
if (rx == 0 && ry == 2)
|
|
|
|
|
{
|
|
|
|
|
if (next_position == FRONT_NEXT)
|
|
|
|
|
{
|
2021-02-14 13:36:38 +08:00
|
|
|
return FORWARD;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
return BACKWARD;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (rx == 0 && ry == 0)
|
|
|
|
|
{
|
|
|
|
|
if (next_position == FRONT_NEXT)
|
|
|
|
|
{
|
|
|
|
|
next_position = BACK_NEXT;
|
|
|
|
|
return BACKWARD;
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
else
|
2021-02-14 13:36:38 +08:00
|
|
|
{
|
|
|
|
|
next_position = FRONT_NEXT;
|
|
|
|
|
return FORWARD;
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (rx == -1 && ry == 1)
|
|
|
|
|
{
|
|
|
|
|
if (next_position == FRONT_NEXT)
|
|
|
|
|
{
|
2021-02-14 13:36:38 +08:00
|
|
|
return FORLEFTWARD;
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2021-02-14 13:36:38 +08:00
|
|
|
next_position = FRONT_NEXT;
|
2021-02-14 12:50:42 +08:00
|
|
|
return BACKLEFTWARD;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else if (rx == 1 && ry == 1)
|
|
|
|
|
{
|
|
|
|
|
if (next_position == FRONT_NEXT)
|
|
|
|
|
{
|
2021-02-14 13:36:38 +08:00
|
|
|
return FORRIGHTWARD;
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2021-02-14 13:36:38 +08:00
|
|
|
next_position = FRONT_NEXT;
|
2021-02-14 12:50:42 +08:00
|
|
|
return BACKRIGHTWARD;
|
|
|
|
|
}
|
|
|
|
|
}
|
2021-02-18 15:13:40 +08:00
|
|
|
else DebugCanNotContinue("CALC DIRECTION"); //调试用
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
//沿线直行,在触发条件后离开函数但不停止
|
2021-02-15 11:01:17 +08:00
|
|
|
void LineForward(uint8_t end_position, float speed_rate)
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
2021-02-15 12:28:14 +08:00
|
|
|
#ifdef TAICHI_DEBUG
|
|
|
|
|
//调试输出沿线直行状态
|
|
|
|
|
Serial.print("#TAICHI: Line Forward");
|
|
|
|
|
Serial.print(" end_position: ");
|
|
|
|
|
Serial.println((int)end_position);
|
|
|
|
|
#endif
|
|
|
|
|
|
2021-02-14 12:50:42 +08:00
|
|
|
//记录开始时间
|
2021-02-18 11:31:28 +08:00
|
|
|
unsigned long begin_time = millis();
|
2021-02-14 12:50:42 +08:00
|
|
|
|
|
|
|
|
while(1)
|
|
|
|
|
{
|
|
|
|
|
if (!sensor.IsWhite(GRAY_3) && sensor.IsWhite(GRAY_4)) //左侧越线
|
|
|
|
|
{
|
|
|
|
|
move.ForRightward(speed_rate, LINE_FIX_SPEED_RATE);
|
|
|
|
|
}
|
|
|
|
|
else if (sensor.IsWhite(GRAY_3) && !sensor.IsWhite(GRAY_4)) //右侧越线
|
|
|
|
|
{
|
|
|
|
|
move.ForLeftward(speed_rate, LINE_FIX_SPEED_RATE);
|
|
|
|
|
}
|
|
|
|
|
else if (sensor.IsWhite(GRAY_3) && sensor.IsWhite(GRAY_4)) //在白线上
|
|
|
|
|
{
|
|
|
|
|
move.Forward(speed_rate);
|
|
|
|
|
}
|
|
|
|
|
else //均不符合,则低速后退,尝试回到白线上
|
|
|
|
|
{
|
2021-02-15 13:24:30 +08:00
|
|
|
move.Backward(LINE_FIND_SPEED_RATE);
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
if (end_position == FRONT_END) //前端接触线离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
|
|
|
|
if (sensor.IsWhite(GRAY_1) && sensor.IsWhite(GRAY_2))
|
|
|
|
|
break;
|
|
|
|
|
}
|
2021-02-18 14:31:51 +08:00
|
|
|
else if (end_position == BACK_END) //后端接触线离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
|
|
|
|
if (sensor.IsWhite(GRAY_5) && sensor.IsWhite(GRAY_6))
|
|
|
|
|
break;
|
|
|
|
|
}
|
2021-02-18 14:31:51 +08:00
|
|
|
else if (end_position == CATCH_END) //到达抓取位置离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
2021-02-18 11:31:28 +08:00
|
|
|
if (millis() - begin_time > CATCH_MOVE_DELAY_TIME)
|
2021-02-14 12:50:42 +08:00
|
|
|
break;
|
|
|
|
|
}
|
2021-02-18 14:31:51 +08:00
|
|
|
else //到达释放位置离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
2021-02-18 11:31:28 +08:00
|
|
|
if (millis() - begin_time > RELEASE_MOVE_DELAY_TIME)
|
2021-02-14 15:44:39 +08:00
|
|
|
break;
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
}
|
2021-02-15 12:28:14 +08:00
|
|
|
|
|
|
|
|
#ifdef TAICHI_DEBUG
|
|
|
|
|
//调试输出沿线直行结束
|
|
|
|
|
Serial.println("#TAICHI: End Line Forward");
|
|
|
|
|
#endif
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
//沿线后退,在触发条件后离开函数但不停止
|
2021-02-15 11:01:17 +08:00
|
|
|
void LineBackward(uint8_t end_position, float speed_rate)
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
2021-02-15 12:28:14 +08:00
|
|
|
#ifdef TAICHI_DEBUG
|
|
|
|
|
//调试输出沿线后退状态
|
|
|
|
|
Serial.print("#TAICHI: Line Backward");
|
|
|
|
|
Serial.print(" end_position: ");
|
|
|
|
|
Serial.println((int)end_position);
|
|
|
|
|
#endif
|
|
|
|
|
|
2021-02-14 12:50:42 +08:00
|
|
|
while(1)
|
|
|
|
|
{
|
|
|
|
|
if (!sensor.IsWhite(GRAY_3) && sensor.IsWhite(GRAY_4)) //左侧越线
|
|
|
|
|
{
|
|
|
|
|
move.BackRightward(speed_rate, LINE_FIX_SPEED_RATE);
|
|
|
|
|
}
|
|
|
|
|
else if (sensor.IsWhite(GRAY_3) && !sensor.IsWhite(GRAY_4)) //右侧越线
|
|
|
|
|
{
|
|
|
|
|
move.BackLeftward(speed_rate, LINE_FIX_SPEED_RATE);
|
|
|
|
|
}
|
|
|
|
|
else if (sensor.IsWhite(GRAY_3) && sensor.IsWhite(GRAY_4)) //在白线上
|
|
|
|
|
{
|
|
|
|
|
move.Backward(speed_rate);
|
|
|
|
|
}
|
|
|
|
|
else //均不符合,则低速前进,尝试回到白线上
|
|
|
|
|
{
|
2021-02-15 13:24:30 +08:00
|
|
|
move.Forward(LINE_FIND_SPEED_RATE);
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
if (end_position == FRONT_END) //前端接触线离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
|
|
|
|
if (sensor.IsWhite(GRAY_1) && sensor.IsWhite(GRAY_2))
|
|
|
|
|
break;
|
|
|
|
|
}
|
2021-02-18 14:31:51 +08:00
|
|
|
else //后端接触线离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
|
|
|
|
if (sensor.IsWhite(GRAY_5) && sensor.IsWhite(GRAY_6))
|
|
|
|
|
break;
|
2021-02-18 14:31:51 +08:00
|
|
|
}
|
2021-02-15 12:28:14 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef TAICHI_DEBUG
|
|
|
|
|
//调试输出沿线后退结束
|
|
|
|
|
Serial.println("#TAICHI: End Line Backward");
|
|
|
|
|
#endif
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2021-02-21 19:28:17 +08:00
|
|
|
//直行或后退或转向,完成后离开函数但不停止。会自动跳过无需前往的释放点
|
2021-02-15 19:17:59 +08:00
|
|
|
void TurnDirection(float speed_rate)
|
2021-02-14 12:50:42 +08:00
|
|
|
{
|
2021-02-21 19:28:17 +08:00
|
|
|
//若下下点为释放点,判断是否需要跳过
|
|
|
|
|
if ((route[next_next_flag][TYPE] == RELEASE_POINT && (!is_claw_catch || is_carry)) || (route[next_next_flag][TYPE] == GETOUT_POINT && !is_carry))
|
|
|
|
|
{
|
|
|
|
|
UpdateFlag(JUMP_DEAD_ROAD);
|
|
|
|
|
|
|
|
|
|
#ifdef TAICHI_DEBUG
|
|
|
|
|
//调试输出直行或后退或转向状态
|
|
|
|
|
Serial.println("#TAICHI: JUMP THE RELEASE/GETOUT POINT FOR NOTHING CARRIED");
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2021-02-15 19:17:59 +08:00
|
|
|
uint8_t direction = CalcDirection();
|
|
|
|
|
|
2021-02-15 12:28:14 +08:00
|
|
|
#ifdef TAICHI_DEBUG
|
|
|
|
|
//调试输出直行或后退或转向状态
|
|
|
|
|
Serial.print("#TAICHI: Turn Direction");
|
|
|
|
|
Serial.print(" direction: ");
|
|
|
|
|
Serial.println((int)direction);
|
|
|
|
|
#endif
|
|
|
|
|
|
2021-02-14 12:50:42 +08:00
|
|
|
if (direction == FORWARD) //继续直行
|
|
|
|
|
{
|
2021-02-18 14:31:51 +08:00
|
|
|
//沿线直行,到后端传感器接触线离开函数
|
2021-02-14 12:50:42 +08:00
|
|
|
LineForward(BACK_END, speed_rate);
|
|
|
|
|
}
|
|
|
|
|
else if (direction == BACKWARD) //继续后退
|
|
|
|
|
{
|
2021-02-18 14:31:51 +08:00
|
|
|
//沿线后退,到前端传感器接触线离开函数
|
2021-02-14 23:38:28 +08:00
|
|
|
LineBackward(FRONT_END, speed_rate);
|
2021-02-14 12:50:42 +08:00
|
|
|
}
|
|
|
|
|
else //继续转向
|
|
|
|
|
{
|
|
|
|
|
move.MoveDirection(direction, speed_rate);
|
2021-02-10 20:13:28 +08:00
|
|
|
|
2021-02-14 12:50:42 +08:00
|
|
|
//传感器判断转向完成
|
|
|
|
|
delay(TRUN_CHECK_DELAY); //延时后判断
|
|
|
|
|
while(!(sensor.IsWhite(GRAY_3) && sensor.IsWhite(GRAY_4))) {}
|
|
|
|
|
}
|
2021-02-15 12:28:14 +08:00
|
|
|
|
|
|
|
|
#ifdef TAICHI_DEBUG
|
|
|
|
|
//调试输出直行或后退或转向结束
|
|
|
|
|
Serial.println("#TAICHI: End Turn Direction");
|
|
|
|
|
#endif
|
2021-02-15 13:14:46 +08:00
|
|
|
}
|
2021-02-17 19:35:43 +08:00
|
|
|
|
|
|
|
|
|
|
|
|
|
//抓取环,并判定是否抓取成功
|
2021-02-18 14:31:51 +08:00
|
|
|
bool CatchAndCheck(float speed)
|
2021-02-17 19:35:43 +08:00
|
|
|
{
|
2021-02-18 13:19:34 +08:00
|
|
|
//记录开始时间
|
|
|
|
|
unsigned long begin_time = millis();
|
|
|
|
|
|
|
|
|
|
//抓取次数
|
|
|
|
|
uint8_t catch_times = 0;
|
|
|
|
|
|
2021-02-17 19:35:43 +08:00
|
|
|
//执行抓取动作
|
|
|
|
|
servo.Catch(speed);
|
2021-02-18 13:19:34 +08:00
|
|
|
catch_times++;
|
2021-02-17 19:35:43 +08:00
|
|
|
|
|
|
|
|
//等待完成动作
|
2021-02-18 13:19:34 +08:00
|
|
|
while (millis() - begin_time < CATCH_DELAY_TIME)
|
2021-02-17 19:35:43 +08:00
|
|
|
{
|
2021-02-18 13:19:34 +08:00
|
|
|
if (sensor.IsPushed(BUTTON_1)) //开关 1 闭合,即爪子两端接触,说明抓取失败
|
2021-02-17 19:35:43 +08:00
|
|
|
{
|
2021-02-17 19:52:59 +08:00
|
|
|
#ifdef TAICHI_DEBUG
|
2021-02-17 19:56:17 +08:00
|
|
|
//调试输出失败信息
|
2021-02-18 13:19:34 +08:00
|
|
|
Serial.print("#TAICHI: **********FAIL CATCH!**********");
|
|
|
|
|
Serial.print(" catch_times: "); Serial.println((int)catch_times);
|
2021-02-17 19:52:59 +08:00
|
|
|
#endif
|
|
|
|
|
|
2021-02-18 13:19:34 +08:00
|
|
|
if (catch_times == MAX_CATCH_TIMES) //达到最大尝试次数,返回
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
|
|
//停止动作组运行
|
|
|
|
|
servo.StopActionGroup();
|
|
|
|
|
|
2021-02-18 13:33:55 +08:00
|
|
|
//打开爪子
|
|
|
|
|
if(!OpenClawAndCheck()) //未能打开爪子
|
2021-02-18 13:19:34 +08:00
|
|
|
return false;
|
2021-02-18 21:37:56 +08:00
|
|
|
|
|
|
|
|
//更新时间
|
|
|
|
|
begin_time = millis();
|
|
|
|
|
|
|
|
|
|
//执行抓取动作
|
|
|
|
|
servo.Catch(speed);
|
|
|
|
|
catch_times++;
|
2021-02-17 19:35:43 +08:00
|
|
|
}
|
|
|
|
|
}
|
2021-02-17 19:52:59 +08:00
|
|
|
|
|
|
|
|
#ifdef TAICHI_DEBUG
|
2021-02-17 19:56:17 +08:00
|
|
|
//调试输出成功信息
|
2021-02-17 19:52:59 +08:00
|
|
|
Serial.println("#TAICHI: SUCCUESS CATCH!");
|
|
|
|
|
#endif
|
|
|
|
|
|
2021-02-18 14:31:51 +08:00
|
|
|
is_claw_catch = true;
|
2021-02-17 19:52:59 +08:00
|
|
|
return true;
|
2021-02-18 13:33:55 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
//打开爪子,并判断爪子是否能正常工作
|
|
|
|
|
bool OpenClawAndCheck(void)
|
|
|
|
|
{
|
|
|
|
|
//记录开始时间
|
|
|
|
|
unsigned long begin_time = millis();
|
|
|
|
|
|
|
|
|
|
//打开爪子
|
|
|
|
|
servo.OpenClaw();
|
|
|
|
|
|
|
|
|
|
//等待完成动作
|
|
|
|
|
while (millis() - begin_time < CLAW_OPEN_USE_TIME)
|
|
|
|
|
{
|
|
|
|
|
if (!sensor.IsPushed(BUTTON_1)) //开关 1 打开,即爪子两端脱离接触,说明打开爪子成功
|
|
|
|
|
{
|
|
|
|
|
#ifdef TAICHI_DEBUG
|
|
|
|
|
//调试输出成功信息
|
|
|
|
|
Serial.println("#TAICHI: SUCCUESS OPEN CLAW!");
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
is_claw_ok = true;
|
|
|
|
|
return true;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef TAICHI_DEBUG
|
|
|
|
|
//调试输出失败信息
|
|
|
|
|
Serial.println("#TAICHI: $$$$$$$$$$FAIL CLAW!$$$$$$$$$$");
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
is_claw_ok = false;
|
|
|
|
|
return false;
|
2021-02-17 19:35:43 +08:00
|
|
|
}
|