#include "math.h" #include "Strategie.h" #include "Geometrie_robot.h" #include "Commande_vitesse.h" #include "Strategie_2024_pots.h" #include "i2c_annexe.h" #include "Localisation.h" float angle_bras[6] = { 180 * DEGRE_EN_RADIAN, 120 * DEGRE_EN_RADIAN, 60 * DEGRE_EN_RADIAN, 0, -60 * DEGRE_EN_RADIAN, -120 * DEGRE_EN_RADIAN }; float angle_bras_correction[6] = { 0 * DEGRE_EN_RADIAN, 0 * DEGRE_EN_RADIAN, 0 * DEGRE_EN_RADIAN, 0, 0 * DEGRE_EN_RADIAN, 7 * DEGRE_EN_RADIAN }; float distance_bras_correction_mm[6] = { 0, 0, -10, -15, 0, 0 }; enum etat_bras_t{ BRAS_LIBRE, BRAS_OCCUPE, } etat_bras[NB_BRAS]; struct position_t position_pots_dans_groupe_pot[5] = { {.x_mm = -40, .y_mm = 69.2, .angle_radian = -60 * DEGRE_EN_RADIAN}, {.x_mm = 40, .y_mm = 69.2, .angle_radian = -120 * DEGRE_EN_RADIAN}, {.x_mm = -80, .y_mm = 0, .angle_radian = -90 * DEGRE_EN_RADIAN}, {.x_mm = 80, .y_mm = 0, .angle_radian = -90 * DEGRE_EN_RADIAN}, {.x_mm = 0, .y_mm = 0, .angle_radian = -90 * DEGRE_EN_RADIAN} }; struct position_t position_groupe_pot[6] = { {.x_mm = 36.1, .y_mm = 1386.8, .angle_radian = -90 * DEGRE_EN_RADIAN}, {.x_mm = 36.1, .y_mm = 616.2, .angle_radian = -90 * DEGRE_EN_RADIAN}, {.x_mm = 1020, .y_mm = 36.4, .angle_radian = 0 * DEGRE_EN_RADIAN}, {.x_mm = 2000, .y_mm = 36.4, .angle_radian = 0 * DEGRE_EN_RADIAN}, {.x_mm = 2963.9, .y_mm = 616.2, .angle_radian = 90 * DEGRE_EN_RADIAN}, {.x_mm = 2963.9, .y_mm = 1386.8, .angle_radian = 90 * DEGRE_EN_RADIAN} }; /// @brief renvoie la position du centre du pot ainsi que l'ange par lequel l'attraper /// @param groupe_pot Position du groupe de pot /// @param num_pot Pot à prendre, entre 0 et 4 (ou utiliser les macros POT_x) struct position_t groupe_pot_get_pot(unsigned int groupe_pot, unsigned int num_pot){ struct position_t position_pot; struct position_t my_position_groupe_pot; my_position_groupe_pot = position_groupe_pot[groupe_pot]; float angle_groupe_pot = my_position_groupe_pot.angle_radian; position_pot.x_mm = my_position_groupe_pot.x_mm + cosf(angle_groupe_pot) * position_pots_dans_groupe_pot[num_pot].x_mm - sinf(angle_groupe_pot) * position_pots_dans_groupe_pot[num_pot].y_mm; position_pot.y_mm = my_position_groupe_pot.y_mm + sinf(angle_groupe_pot) * position_pots_dans_groupe_pot[num_pot].x_mm + cosf(angle_groupe_pot) * position_pots_dans_groupe_pot[num_pot].y_mm; position_pot.angle_radian = position_pots_dans_groupe_pot[num_pot].angle_radian + angle_groupe_pot; return position_pot; } /// @brief A Affiner /// @return numéro du bras libre (entre 0 et 5) int ordre_bras[NB_BRAS]={5, 0, 1, 2, 3, 4}; int get_bras_libre(void){ for (int i=0; i<NB_BRAS; i++){ if(etat_bras[ordre_bras[i]] == BRAS_LIBRE){ return ordre_bras[i]; } } return 9; } int ordre_pot[5]={POT_1, POT_2, POT_4, POT_5, POT_3}; int get_pot_suivant(int pot){ for(int i=0; i<4; i++){ if(ordre_pot[i] == pot){ return ordre_pot[++i]; } } return POT_INVALIDE; } /// @brief Fonction qui déplace le robot jusqu'à la zone pour attraper les pots et qui attrape les 5 pots enum etat_action_t Strat_2024_attrape_pot(unsigned int groupe_pot, uint32_t step_ms){ // Parcourir la trajectoire pour aller jusqu'au premier pot static struct position_t position_pot, position_approche_pot, position_attrape_pot; enum etat_action_t etat_action; enum validite_vl53l8_t validite; float distance, angle; static int bras, tempo_ms; // Pour le 1er pot static int pot = POT_1; static enum { AP_ALLER_VERS_GROUPE_POT, AP_RECALE, AP_ORIENTE, AP_APPROCHE_POT, AP_ATTRAPE_POT, AP_RETOUR_ET_LEVE_POT, AP_FINALISE } etat_attrape_pot = AP_ALLER_VERS_GROUPE_POT; // Pour chaque pot // Baisser le bras correspondant // Aller jusqu'au point de prise de pot en s'orientant pour prendre le pot // Avancer de X cm en direction du pot // Reculer dans l'axe de prise et rejoindre le point de prise suivant // Pendant le mouvement, apres 1 sec (à confirmer) Lever le bras switch (etat_attrape_pot) { case AP_ALLER_VERS_GROUPE_POT: bras = get_bras_libre(); position_pot = groupe_pot_get_pot(groupe_pot, POT_1); position_approche_pot = Geometrie_deplace(position_pot, -DISTANCE_APPROCHE_POT_MM); position_attrape_pot = Geometrie_deplace(position_pot, -(DISTANCE_ATTRAPE_POT_MM + distance_bras_correction_mm[bras])); Trajet_config(TRAJECT_CONFIG_AVANCE_ET_TOURNE); /*etat_action = Strategie_tourner_et_aller_a( position_approche_pot.x_mm, position_approche_pot.y_mm, position_approche_pot.angle_radian - angle_bras[0], SANS_EVITEMENT, step_ms);*/ etat_action = Strategie_tourner_et_aller_a( position_approche_pot.x_mm, position_approche_pot.y_mm, (-30. *DEGRE_EN_RADIAN), EVITEMENT_SANS_EVITEMENT, step_ms); if (etat_action == ACTION_TERMINEE){ etat_attrape_pot = AP_RECALE; i2c_annexe_set_mode_VL53L8(VL53L8_DISTANCE_LOIN); } break; case AP_RECALE: i2c_annexe_get_VL53L8(&validite, &angle, &distance); if(validite == VL53L8_DISTANCE_LOIN){ if(fabs(distance + DISTANCE_CENTRE_CAPTEUR - Localisation_get().x_mm) < 25){ i2c_annexe_set_mode_VL53L8(VL53L8_INVALIDE); commande_vitesse_stop(); //if(couleur == COULEUR_BLEU){ // Localisation_set_x(distance + DISTANCE_CENTRE_CAPTEUR); /*}else{ Localisation_set_x(3000 - (distance + DISTANCE_CENTRE_CAPTEUR)); }*/ etat_attrape_pot = AP_ORIENTE; }else{ //printf("Erreur - recalage trop loin\n"); } } break; case AP_ORIENTE: bras = get_bras_libre(); if(Strategie_tourner_a(position_approche_pot.angle_radian - angle_bras[bras], step_ms) == ACTION_TERMINEE){ etat_attrape_pot = AP_ATTRAPE_POT; } break; case AP_APPROCHE_POT: bras = get_bras_libre(); Trajet_config(TRAJECT_CONFIG_AVANCE_ET_TOURNE); /*etat_action = Strategie_tourner_et_aller_a( position_approche_pot.x_mm, position_approche_pot.y_mm, position_approche_pot.angle_radian - angle_bras[bras], SANS_EVITEMENT, step_ms);*/ etat_action = Strategie_aller_a_puis_tourner( position_approche_pot.x_mm, position_approche_pot.y_mm, position_approche_pot.angle_radian - angle_bras[bras], EVITEMENT_SANS_EVITEMENT, step_ms); if (etat_action == ACTION_TERMINEE){ position_attrape_pot = Geometrie_deplace(position_pot, -(DISTANCE_ATTRAPE_POT_MM + distance_bras_correction_mm[bras])); etat_attrape_pot = AP_ATTRAPE_POT; } break; case AP_ATTRAPE_POT: Trajet_config(TRAJECT_CONFIG_AVANCE_ET_TOURNE); i2c_annexe_actionneur_pot(bras, BRAS_POT_SOL, DOIGT_TIENT); etat_action = Strategie_tourner_et_aller_a( position_attrape_pot.x_mm, position_attrape_pot.y_mm, position_attrape_pot.angle_radian - angle_bras[bras], EVITEMENT_SANS_EVITEMENT, step_ms); if (etat_action == ACTION_TERMINEE){ tempo_ms=250; etat_attrape_pot = AP_RETOUR_ET_LEVE_POT; etat_bras[bras] = BRAS_OCCUPE; } break; case AP_RETOUR_ET_LEVE_POT: if(tempo_ms >= 0){ tempo_ms -= step_ms; }else{ i2c_annexe_actionneur_pot(bras, BRAS_HAUT, DOIGT_TIENT); } etat_action = Strategie_tourner_et_aller_a( position_approche_pot.x_mm, position_approche_pot.y_mm, position_attrape_pot.angle_radian - angle_bras[bras], EVITEMENT_SANS_EVITEMENT, step_ms); if (etat_action == ACTION_TERMINEE){ etat_attrape_pot = AP_APPROCHE_POT; pot = get_pot_suivant(pot); if(pot > 4){ tempo_ms=250; etat_attrape_pot = AP_FINALISE; break; } position_pot = groupe_pot_get_pot(groupe_pot, pot); position_approche_pot = Geometrie_deplace(position_pot, -DISTANCE_APPROCHE_POT_MM); position_attrape_pot = Geometrie_deplace(position_pot, -DISTANCE_ATTRAPE_POT_MM); } break; case AP_FINALISE: if(tempo_ms >= 0){ tempo_ms -= step_ms; }else{ i2c_annexe_actionneur_pot(bras, BRAS_HAUT, DOIGT_TIENT); } etat_action = Strategie_tourner_et_aller_a( position_approche_pot.x_mm, position_approche_pot.y_mm, position_attrape_pot.angle_radian - angle_bras[bras], EVITEMENT_SANS_EVITEMENT, step_ms); if (etat_action == ACTION_TERMINEE){ etat_attrape_pot = AP_ALLER_VERS_GROUPE_POT; return ACTION_TERMINEE; } break; default: break; } return ACTION_EN_COURS; }