CHEWY Robot Arm

4 min read
C++SolidworksKinematics AlgorithmsArduino

Introduction

I designed a 4 revolute joint robot arm and wrote forward kinematic algorithms in C++ to display the end effector position relative to the base (1st joint) and inverse kinematic algorithms in Python that allows us to input a desired position and in return fetch the required joint angles to reach that position.

I designed the base and arms in SolidWorks and converted the model to .gltf so that I could use the Three.js library along with react-three-fiber to display it above. Regular panning, zooming, and rotating controls work in the outlined canvas.

Features:

  • LCD Screen Displaying
    • xyz coordinates of end effector
    • q1q_1, q2q_2, q3q_3, q4q_4 joint angles
  • Simple Potentiometer controls
  • Ability to grab small/lightweight objects

Improvements to be made:

  • Better power supply.
    • When motors were drawing current under heavier loads they would struggle a lot. A better power supply design is definitely needed to smoothen out these spikes.
  • More integrated algorithm and UI
    • The NumPy python library was used to write algorithms for inverse kinematics. So the desired end effector position would be inputed in the python script, required joint angles would be manually inputed using Potentiometters and the Robot would move there. A more integrated yet simple UI such as a number pad for each xyz coordinate would definitely be an improvement to the project.

C++/Arduino Code

Library includes and variable set up:

#include <LiquidCrystal.h>
#include <Servo.h>
#include <Math.h>

LiquidCrystal lcd(13, 12, 7, 4, 3, 2);
Servo servos[5];
int const numOfServos = 5;

int const pot1Pin = A0;
int const pot2Pin = A1;
int const pot3Pin = A2;
int const pot4Pin = A3;
int const pot5Pin = A4;

int potValues[5];
int angles[5];
int dh_angles[4];
const double pi = 3.1415927;

double const d1 = 42.25;
double const a2 = 100.0;
double const a3 = 81.4;
double const a4 = 125.0;
double x_ee(double q1, double q2, double q3, double q4);
double y_ee(double q1, double q2, double q3, double q4);
double z_ee(double q1, double q2, double q3, double q4);


#include <LiquidCrystal.h>
#include <Servo.h>
#include <Math.h>

LiquidCrystal lcd(13, 12, 7, 4, 3, 2);
Servo servos[5];
int const numOfServos = 5;

int const pot1Pin = A0;
int const pot2Pin = A1;
int const pot3Pin = A2;
int const pot4Pin = A3;
int const pot5Pin = A4;

int potValues[5];
int angles[5];
int dh_angles[4];
const double pi = 3.1415927;

double const d1 = 42.25;
double const a2 = 100.0;
double const a3 = 81.4;
double const a4 = 125.0;
double x_ee(double q1, double q2, double q3, double q4);
double y_ee(double q1, double q2, double q3, double q4);
double z_ee(double q1, double q2, double q3, double q4);

void setup() {
  lcd.begin(16, 2);
  servos[0].attach(11);
  servos[1].attach(10);
  servos[2].attach(9);
  servos[3].attach(6);
  servos[4].attach(5);

  Serial.begin(9600);
}

void loop() {
  // read in potentiometer values
  potValues[0] = analogRead(pot1Pin);
  potValues[1] = analogRead(pot2Pin);
  potValues[2] = analogRead(pot3Pin);
  potValues[3] = analogRead(pot4Pin);
  potValues[4] = analogRead(pot5Pin);

  // angle values for servos
  for (int i(0); i < numOfServos - 1; i++)
  {
    angles[i] = map(potValues[i], 0, 1023, 0, 180);
  }
  // for claw servo
  angles[4] = map(potValues[4], 0, 1023, 0, 65);

  // DH joint angles
  dh_angles[0]= angles[0];
  dh_angles[1] = angles[1];
  dh_angles[2] = map(angles[2], 0, 180, 90, -90);
  dh_angles[3] = map(angles[3], 0, 180, -90, 90);

  // printing servo angles and dh angles to Serial monitor
  for(int i(0); i < numOfServos-1; i++)
  {
    Serial.print(dh_angles[i]);
    Serial.print("\t");
  }
  for(int i(0); i < numOfServos-1; i++)
  {
    Serial.print(angles[i]);
    Serial.print("\t");
    if (i == 3)
    {
      Serial.println("");
    }
  }

  // printing dh angles to LCD
  /*
  lcd.setCursor(0, 0);
  lcd.print("dh joint angles: ");
  lcd.setCursor(0, 1);
  for (int i(0); i < numOfServos-1; i++)
  {
    lcd.print(dh_angles[i]);
    lcd.print(" ");
  }
  */

  // printing xyz coordinates relative to base frame to LCD
  int x_ee_int = x_ee(dh_angles[0],dh_angles[1],dh_angles[2],dh_angles[3]);
  int y_ee_int = y_ee(dh_angles[0],dh_angles[1],dh_angles[2],dh_angles[3]);
  int z_ee_int = z_ee(dh_angles[0],dh_angles[1],dh_angles[2],dh_angles[3]);
  lcd.setCursor(0, 0);
  lcd.print("xyz coordinates:");
  lcd.setCursor(0, 1);
  lcd.print(x_ee_int);
  lcd.print(" ");
  lcd.print(y_ee_int);
  lcd.print(" ");
  lcd.print(z_ee_int);
  lcd.print(" ");

  // writing to servos
  for(int i(0); i < numOfServos; i++)
  {
    servos[i].write(angles[i]);
  }
  delay(15);
}

double x_ee(double q1, double q2, double q3, double q4)
{
  q1 = q1 * (pi/180);
  q2 = q2 * (pi/180);
  q3 = q3 * (pi/180);
  q4 = q4 * (pi/180);
  return a2*cos(q1)*cos(q2)-a3*sin(q2)*sin(q3)*cos(q1)+a3*cos(q1)*cos(q2)*cos(q3)+a4*(-sin(q2)*sin(q3)*cos(q1)+cos(q1)*cos(q2)*cos(q3))*cos(q4)+
    a4*(-sin(q2)*cos(q1)*cos(q3)-sin(q3)*cos(q1)*cos(q2))*sin(q4);
}
double y_ee(double q1, double q2, double q3, double q4)
{
  q1 = q1 * (pi/180);
  q2 = q2 * (pi/180);
  q3 = q3 * (pi/180);
  q4 = q4 * (pi/180);
  return a2*sin(q1)*cos(q2)-a3*sin(q1)*sin(q2)*sin(q3)+a3*sin(q1)*cos(q2)*cos(q3)+a4*(-sin(q1)*sin(q2)*sin(q3)+sin(q1)*cos(q2)*cos(q3))*cos(q4)+
    a4*(-sin(q1)*sin(q2)*cos(q3)-sin(q1)*sin(q3)*cos(q2))*sin(q4);
}
double z_ee(double q1, double q2, double q3, double q4)
{
  q1 = q1 * (pi/180);
  q2 = q2 * (pi/180);
  q3 = q3 * (pi/180);
  q4 = q4 * (pi/180);
  return a2*sin(q2)+a3*sin(q2)*cos(q3)+a3*sin(q3)*cos(q2)+a4*(-sin(q2)*sin(q3)+cos(q2)*cos(q3))*sin(q4)+a4*(sin(q2)*cos(q3)+sin(q3)*cos(q2))*cos(q4)+d1;
}