Sommario:
2025 Autore: John Day | [email protected]. Ultima modifica: 2025-01-23 14:49
Questo progetto riguarda l'implementazione di uno schizzo Arduino breve e relativamente semplice per fornire il posizionamento cinematico inverso XYZ. Avevo costruito un braccio robotico con 6 servocomandi, ma quando si trattava di trovare un software per eseguirlo, non c'era molto là fuori tranne che per programmi personalizzati in esecuzione su schermi servo personalizzati come l'SSC-32 (U) o altri programmi e app che erano complicato da installare e comunicare con il braccio. Poi ho trovato l'eccellente "Robotic Arm Inverse Kinematics on Arduino" di Oleg Mazurov, dove ha implementato la cinematica inversa in un semplice sketch Arduino.
Ho apportato due modifiche per adattare il suo codice:
1. Ho usato la libreria VarSpeedServo al posto della sua libreria servo shield personalizzata perché potrei quindi controllare la velocità dei servi e non avrei dovuto usare la servo shield che ha usato. Per chiunque stia pensando di eseguire il codice fornito qui, consiglio di utilizzare questa libreria VarSpeedServo, piuttosto che la libreria servo.h, in modo da poter rallentare il movimento del braccio robotico durante lo sviluppo o potresti scoprire che il braccio ti colpirà inaspettatamente la faccia o peggio perché si muoverà alla massima velocità del servo.
2. Uso un semplice sensore/servo shield per collegare i servi all'Arduino Uno, ma non richiede una libreria servo speciale poiché utilizza solo i pin di Arduino. Costa solo pochi euro ma non è obbligatorio. Fa una bella connessione pulita dei servi all'Arduino. E non tornerò mai più ai servi cablati su Arduino Uno ora. Se utilizzi questo scudo sensore/servo, devi apportare una piccola modifica che illustrerò di seguito.
Il codice funziona alla grande e permette di azionare il braccio utilizzando un'unica funzione in cui si passano i parametri x, y, x e velocità. Per esempio:
set_arm(0, 240, 100, 0, 20); // i parametri sono (x, y, z, angolo di presa, velocità del servo)
ritardo (3000); // è necessario un ritardo per consentire il tempo di attivazione per spostarsi in questa posizione
Non potrebbe essere più semplice. Includerò lo schizzo qui sotto.
Il video di Oleg è qui: Controllo del braccio robotico con Arduino e mouse USB
Il programma, le descrizioni e le risorse originali di Oleg: la cinematica inversa di Oleg per Arduino Uno
Non capisco tutta la matematica dietro la routine, ma la cosa bella è che non devi usare il codice. Spero che ci proverai.
Passaggio 1: modifiche hardware
1. L'unica cosa che è richiesta è che il tuo servo giri nelle direzioni previste, il che potrebbe richiedere di invertire fisicamente il montaggio dei tuoi servi. Vai a questa pagina per vedere la direzione del servo prevista per i servi di base, spalla, gomito e polso:
2. Se usi lo scudo del sensore che sto usando, devi fare una cosa: piegare il pin che collega il 5v dallo scudo all'Arduino Uno in modo che non si colleghi alla scheda Uno. Vuoi usare la tensione esterna sullo scudo per alimentare solo i tuoi servi, non l'Arduino Uno o potrebbe distruggere l'Uno, so che ho bruciato due schede Uno quando la mia tensione esterna era di 6 volt anziché 5. Questo ti consente usare più di 5 V per alimentare i tuoi servi, ma se la tua tensione esterna è superiore a 5 volt, non collegare alcun sensore da 5 volt allo scudo o saranno fritti.
Passaggio 2: scarica la libreria VarSpeedServo
Devi usare questa libreria che sostituisce la libreria servo standard di arduino perché ti consente di passare una velocità del servo nell'istruzione di scrittura del servo. La biblioteca si trova qui:
Libreria VarSpeedServo
Puoi semplicemente usare il pulsante zip, scaricare il file zip e quindi installarlo con l'IDE di Arduino. Una volta installato il comando nel tuo programma sarà simile a: servo.write(100, 20);
Il primo parametro è l'angolo e il secondo è la velocità del servo da 0 a 255 (piena velocità).
Passaggio 3: esegui questo schizzo
Ecco il programma del concorso. Devi modificare alcuni parametri per le dimensioni del tuo braccio robotico:
1. Lunghezze BASE_HGT, HUMERUS, ULNA, PINZA in millimetri.
2. Inserisci i numeri dei pin del tuo servo
3. Immettere il servo min e max nelle istruzioni allegate.
4. Quindi prova un semplice comando set_arm() e poi le funzioni zero_x(), line() e circle() per il test. Assicurati che la velocità del tuo servo sia bassa la prima volta che esegui queste funzioni per evitare di danneggiare il tuo braccio e il tuo braccio.
Buona fortuna.
#include VarSpeedServo.h
/* Servocomando per braccio AL5D */
/* Dimensioni braccio (mm) */
#define BASE_HGT 90 //altezza della base
#define HUMERUS 100 //"osso" da spalla a gomito
#define ULNA 135 //"osso" dal gomito al polso
#define GRIPPER 200 //pinza (incl. meccanismo di rotazione del polso per impieghi gravosi) lunghezza"
#define ftl(x) ((x)>=0?(long)((x)+0.5):(long)((x)-0.5)) //float alla conversione lunga
/* Nomi/numeri servo *
* Servo di base HS-485HB */
#define BAS_SERVO 4
/* Servo a spalla HS-5745-MG */
#define SHL_SERVO 5
/* Servo a gomito HS-5745-MG */
#define ELB_SERVO 6
/* Servo da polso HS-645MG */
#define WRI_SERVO 7
/* Servo rotazione polso HS-485HB */
#define WRO_SERVO 8
/* Servo pinza HS-422 */
#define GRI_SERVO 9
/* calcoli preliminari */
float hum_sq = OMERO*OMERO;
float uln_sq = ULNA*ULNA;
int servoVelocità = 10;
//servi ServoShield; //Oggetto ServoShield
VarSpeedServo servo1, servo2, servo3, servo4, servo5, servo6;
int loopCounter=0;
int PulseWidth = 6.6;
int microsecondiToDegrees;
configurazione nulla()
{
servo1.attach(BAS_SERVO, 544, 2400);
servo2.attach(SHL_SERVO, 544, 2400);
servo3.attach(ELB_SERVO, 544, 2400);
servo4.attach(WRI_SERVO, 544, 2400);
servo5.attach(WRO_SERVO, 544, 2400);
servo6.attach(GRI_SERVO, 544, 2400);
ritardo (5500);
//servo.start(); //Avvia lo scudo del servo
servo_park();
ritardo (4000);
Serial.begin(9600);
Serial.println("Avvio");
}
ciclo vuoto()
{
contatoreciclo +=1;
//set_arm(-300, 0, 100, 0, 10); //
//ritardo(7000);
//zero_x();
//linea();
//cerchio();
ritardo (4000);
if (contaloop > 1) {
servo_park();
//set_arm(0, 0, 0, 0, 10); // parco
ritardo (5000);
uscita (0); }//pausa programma - premi reset per continuare
//uscita(0);
}
/* routine di posizionamento del braccio che utilizza la cinematica inversa */
/* z è l'altezza, y è la distanza dal centro della base verso l'esterno, x è da un lato all'altro. y, z possono essere solo positivi */
//void set_arm(uint16_t x, uint16_t y, uint16_t z, uint16_t grip_angle)
void set_arm(float x, float y, float z, float grip_angle_d, int servoSpeed)
{
float grip_angle_r = radianti(grip_angle_d); //angolo di presa in radianti per l'uso nei calcoli
/* Angolo base e distanza radiale dalle coordinate x, y */
float bas_angle_r = atan2(x, y);
float rdist = sqrt((x * x) + (y * y));
/* rdist è la coordinata y per il braccio */
y = rdist;
/* Scostamenti grip calcolati in base all'angolo grip */
galleggiante grip_off_z = (sin(grip_angle_r)) * PINZA;
galleggiante grip_off_y = (cos(grip_angle_r)) * PINZA;
/* Posizione del polso */
float wrist_z = (z - grip_off_z) - BASE_HGT;
float wrist_y = y - grip_off_y;
/* Distanza spalla-polso (AKA sw) */
float s_w = (wrist_z * wrist_z) + (wrist_y * wrist_y);
float s_w_sqrt = sqrt(s_w);
/* s_w angolo rispetto al suolo */
float a1 = atan2(wrist_z, wrist_y);
/* s_w angolo rispetto all'omero */
float a2 = acos(((hum_sq - uln_sq) + s_w) / (2 * HUMERUS * s_w_sqrt));
/* angolo della spalla */
float shl_angle_r = a1 + a2;
float shl_angle_d = gradi(shl_angle_r);
/* angolo del gomito */
float elb_angle_r = acos((hum_sq + uln_sq - s_w) / (2 * HUMERUS * ULNA));
float elb_angle_d = gradi(elb_angle_r);
float elb_angle_dn = -(180.0 - elb_angle_d);
/* angolo del polso */
float wri_angle_d = (grip_angle_d - elb_angle_dn) - shl_angle_d;
/* Servo impulsi */
float bas_servopulse = 1500.0 - ((degrees(bas_angle_r)) * pulseWidth);
float shl_servopulse = 1500.0 + ((shl_angle_d - 90.0) * pulseWidth);
float elb_servopulse = 1500.0 - ((elb_angle_d - 90.0) * pulseWidth);
//float wri_servopulse = 1500 + (wri_angle_d * pulseWidth);
//float wri_servopulse = 1500 + (wri_angle_d * pulseWidth);
float wri_servopulse = 1500 - (wri_angle_d * pulseWidth); // aggiornato 2018/2/11 da jimrd - ho cambiato il più in un meno - non sono sicuro di come funzionasse questo codice per chiunque prima. Potrebbe essere che il servo del gomito sia stato montato con 0 gradi rivolto verso il basso anziché verso l'alto.
/* Imposta i servi */
//servos.setposition(BAS_SERVO, ftl(bas_servopulse));
microsecondiToDegrees = map(ftl(bas_servopulse), 544, 2400, 0, 180);
servo1.write(microsecondsToDegrees, servoSpeed); // usa questa funzione in modo da poter impostare la velocità del servo //
//servos.setposition(SHL_SERVO, ftl(shl_servopulse));
microsecondiToDegrees = map(ftl(shl_servopulse), 544, 2400, 0, 180);
servo2.write(microsecondsToDegrees, servoSpeed);
//servos.setposition(ELB_SERVO, ftl(elb_servopulse));
microsecondiToDegrees = map(ftl(elb_servopulse), 544, 2400, 0, 180);
servo3.write(microsecondsToDegrees, servoSpeed);
//servos.setposition(WRI_SERVO, ftl(wri_servopulse));
microsecondiToDegrees = map(ftl(wri_servopulse), 544, 2400, 0, 180);
servo4.write(microsecondsToDegrees, servoSpeed);
}
/* sposta i servi in posizione di parcheggio */
void servo_park()
{
//servos.setposition(BAS_SERVO, 1500);
servo1.write(90, 10);
//servos.setposition(SHL_SERVO, 2100);
servo2.write(90, 10);
//servos.setposition(ELB_SERVO, 2100);
servo3.write(90, 10);
//servos.setposition(WRI_SERVO, 1800);
servo4.write(90, 10);
//servos.setposition(WRO_SERVO, 600);
servo5.write(90, 10);
//servos.setposition(GRI_SERVO, 900);
servo6.write(80, 10);
Restituzione;
}
vuoto zero_x()
{
for(asse y doppio = 250.0; asse y < 400.0; asse y += 1) {
Serial.print(" yaxis=: ");Serial.println(yaxis);
set_arm(0, yaxis, 200.0, 0, 10);
ritardo(10);
}
for(doppio asse y = 400,0; asse y > 250,0; asse y -= 1) {
set_arm(0, yaxis, 200.0, 0, 10);
ritardo(10);
}
}
/* muove il braccio in linea retta */
riga vuota()
{
for(asse x doppio = -100,0; asse x < 100,0; asse x += 0,5) {
set_arm(asse x, 250, 120, 0, 10);
ritardo(10);
}
for(float asse x = 100,0; asse x > -100,0; asse x -= 0,5) {
set_arm(asse x, 250, 120, 0, 10);
ritardo(10);
}
}
cerchio vuoto()
{
#define RAGGIO 50.0
//angolo flottante = 0;
galleggiante zaxis, yaxis;
for(angolo flottante = 0.0; angolo < 360.0; angolo += 1.0) {
yaxis = RAGGIO * sin(radianti(angolo)) + 300;
zaxis = RAGGIO * cos(radianti(angolo)) + 200;
set_arm(0, yaxis, zaxis, 0, 50);
ritardo(10);
}
}
Passaggio 4: fatti, problemi e simili…
1. Quando eseguo la subroutine circle(), il mio robot si muove più in una forma ellittica che in un cerchio. Penso che sia perché i miei servi non sono calibrati. Ne ho testato uno e 1500 microsecondi non erano gli stessi di 90 gradi. Lavorerà su questo per cercare di trovare una soluzione. Non credo che ci sia qualcosa di sbagliato nell'algoritmo ma piuttosto nelle mie impostazioni. Aggiornamento 2018/2/11: ho appena scoperto che ciò è dovuto a un errore nel codice originale. Non vedo come funzionava il suo programma Codice fisso usando questo: float wri_servopulse = 1500 - (wri_angle_d * pulseWidth); (il codice originale stava aggiungendo)
2. Dove posso trovare maggiori informazioni su come funziona la funzione set_arm(): il sito Web di Oleg Mazurov spiega tutto o fornisce collegamenti per maggiori informazioni:
3. Esiste un controllo delle condizioni al contorno? No. Quando al mio braccio robotico viene passata una coordinata xyz non valida, fa questo strano tipo di movimento inarcato come un gatto che si stira. Credo che Oleg controlli il suo ultimo programma che usa una USB per programmare i movimenti del braccio. Guarda il suo video e collegalo al suo ultimo codice.
4. Il codice deve essere ripulito e il codice in microsecondi può essere eliminato.
Consigliato:
Controllo del braccio robotico con TLV493D, joystick e Arduino: 3 passaggi
Controllo del braccio robotico con TLV493D, Joystick e Arduino: un controller alternativo per il tuo robot con un sensore TLV493D, un sensore magnetico con 3 gradi di libertà (x,y,z) con questi puoi controllare i tuoi nuovi progetti con la comunicazione I2C sul tuo microcontrollori e scheda elettronica che Bast P
L'arrivo del braccio robotico intelligente: 3 passaggi
L'arrivo del braccio robotico intelligente: stringere la mano agli ospiti, parlare di cose, mangiare e così via, queste cose ordinarie, perché la salute della nostra vita è nelle cose ordinarie, ma per alcune persone speciali è un sogno. Alcune persone speciali da me menzionate sono le persone disabili che hanno perso
Gioco del braccio robotico - Controller per smartphone: 6 passaggi
Gioco del braccio robotico - Controller per smartphone: Ciao! Ecco un divertente gioco estivo: Il braccio robotico controllato dallo smartphone !! Come puoi vedere nel video, puoi controllare il braccio con alcuni joystick sul tuo smartphone. Puoi anche salvare un modello, che il robot riprodurrà in loop, per
Dimostrazione del braccio robotico del reattore Arbotix e della Pixycam: 11 passaggi
Arbotix Reactor Robot Arm e dimostrazione di Pixycam: siamo 2 studenti del college UCN in Danimarca. Siamo stati incaricati di fare un imperscrutabile come parte della nostra valutazione per la nostra classe, robot e visione. I requisiti del progetto erano di includere uno o più robot di arbotix ed eseguire un compito.Prog
Consigli sulla tecnica e il posizionamento del microfono per il cantante: 5 passaggi
Consigli sulla tecnica del microfono e sul posizionamento per il cantante: Per gli inesperti, l'uso di un microfono può sembrare inizialmente un'operazione abbastanza facile. Devi semplicemente parlare o cantare nella parte superiore in alto e un suono meravigliosamente chiaro ed equilibrato verrà emesso dagli altoparlanti con ampi consensi dall'a