You are currently viewing Building a Line Following Robot with TCS34725 RGB Sensor
This is the 3D-rendered design of the Arduino path-following robot equipped with TCS34725 sensors for accurate line tracking.

Building a Line Following Robot with TCS34725 RGB Sensor

In this tutorial, we’ll upgrade the KY-033 line-following robot from the previous tutorial by replacing the KY-033 IR sensor with a TCS34725 RGB sensor. This new approach, combined with PID control, improves accuracy and stability, making it ideal for navigating complex tracks with smoother adjustments. Building on the previous project, you’ll gain a deeper insight into robotics and sensor integration while exploring advanced techniques for line-following robots.

A 3D-rendered line-following robot designed with an orange and blue chassis, featuring OmArTronics branding, two wheels, and a support caster wheel, suitable for DIY robotics projects.

Components Needed:

3D-Printed Parts

  • Chassis: Houses the motors, sensors, and Arduino board.
  • Sensor Holder: Aligns the sensors for precise line detection.
  • Arduino Holder: Mounts the Arduino securely to the chassis.

Motors and Wheels

  • 2 DC Motors with Gearboxes: Drive the robot.
  • 2 Wheels: Provide mobility.
  • 1 Support Wheel: Balances the robot for smooth movement.

Electronics

  • Arduino Uno: Microcontroller for programming and control.
  • Arduino Motor Shield Rev3: Drives the motors and distributes power.
  • TCS34725 RGB Light Sensor: Measures light intensity to differentiate between colors.

Accessories

  • Jumper Wires: For connecting components.
  • Screws and Inserts: To securely mount parts.
  • Battery (e.g., 7.4V LiPo): Powers the robot.
  • Power Switch: Optional, for easy on/off control.

Step 1: Designing the Robot in Autodesk Inventor

We started by designing the robot in Autodesk Inventor. The design ensures all components fit securely while maintaining a lightweight and compact structure. Key features include:

  1. Chassis: A sturdy base for mounting the motors, sensors, and Arduino.
  2. Sensor Holder: Holds the TCS34725 sensor for precise line detection.
  3. Arduino Holder: Provides easy access to the Arduino for programming and wiring.

You can download the STL files for these 3D-printed parts from Cults3D.

Step 2: 3D Printing the Components

After finalizing the design, we used a Creality Ender 3 printer to 3D print the components. Materials like PLA or ABS are recommended for durability. Ensure the parts are cleaned and prepared before assembly.

Step 3: Assembling the Line-Following Robot With TCS34725 RGB Sensor

1. Mounting the Motors and Wheels

  • Secure the motors to the chassis using screws and inserts.
  • Attach the wheels to the motor shafts and ensure smooth rotation.
  • Mount the support wheel at the rear for balance.

2. Installing the Electronics

  • Mount the Arduino Uno and Motor Shield to the chassis.
  • Attach the sensors to the sensor holder.
  • Connect the Battery and secure all components.

3. Wiring the Components

  • Wire the motors to the Motor Shield.
  • Connect the sensors to the Arduino pins.
  • Ensure the battery is safely connected and add a switch for convenience.
Wiring diagram for a line-following robot using Arduino Uno, motor shield, DC motors, sensors, and a 9V battery.

Step 4: Understanding PID Control

The TCS34725 RGB sensor is highly effective for distinguishing the black line from the white background due to its ability to measure light intensity across various wavelengths. This approach includes a calibration phase and uses a PID controller to ensure smooth and accurate path following.

Understanding PID Control Fully in the Context of Line Following

The PID (Proportional-Integral-Derivative) controller is a fundamental concept in control systems, designed to adjust outputs based on feedback from a process variable. In the context of the line-following robot, PID ensures dynamic control of motor speeds, enabling precise and smooth path following.

Diagram illustrating a PID feedback control loop with setpoint, controller, plant, and disturbances.

Components of the PID Controller

  1. Proportional (P) Component

    • Purpose: Reacts immediately to the error.
    • Equation:
      y(t) = Kp * e(t)
    • How it works:
      The proportional term multiplies the current error (e) by a proportional gain (Kp). This provides a correction that is directly proportional to the deviation from the desired position, such as the center of the line.
    • Effect:
      • Quick responsiveness to errors.
      • Larger Kp values result in faster reactions but can cause overshooting or instability.
    • Example in Line Following:
      If the robot veers off the line, the proportional component immediately adjusts motor speeds to steer it back.
  2. Integral (I) Component

    • Purpose: Compensates for accumulated errors over time.
    • Equation:
      y(t) = Ki * ∫e(t) dt
    • How it works:
      The integral term sums the error over time and multiplies it by the integral gain (Ki). This helps to correct long-term drift caused by persistent small errors.
    • Effect:
      • Eliminates steady-state error.
      • Slower to react compared to the proportional component.
    • Example in Line Following:
      If the robot consistently drifts to one side, the integral component adjusts the motors to correct the bias over time.
  3. Derivative (D) Component (optional for this project)

    • Purpose: Reacts to the rate of change of the error.
    • Equation:
      y(t) = Kd * (de(t)/dt)
    • How it works:
      The derivative term predicts future errors based on the rate at which the error changes and multiplies this rate by the derivative gain (Kd).
    • Effect:
      • Reduces oscillations and improves stability.
      • Particularly useful for systems experiencing rapid changes.
    • Example in Line Following:
      If the robot oscillates around the line, the derivative component stabilizes it by reducing abrupt motor speed changes.

Combining P, I, and D in a PID Controller

A PID controller combines these three components to provide precise and stable control. The combined equation is:

y(t) = Kp * e(t) + Ki * ∫e(t) dt + Kd * (de(t)/dt)

How it applies to Line Following:

  • The proportional term reacts immediately to how far the robot is from the line.
  • The integral term addresses consistent drift over time.
  • The derivative term dampens oscillations and stabilizes movement.

Dynamic Error Correction with PID

  1. Error Calculation:

    • The error (e) is the difference between the sensor reading and the threshold (middle value).
    • This error drives the PID calculations.
  2. Correction Value (y):

    • The PID controller calculates y to determine how much to adjust motor speeds:
      • speedLeft = baseSpeed + y
      • speedRight = baseSpeed - y
  3. Motor Speed Adjustment:

    • Motor speeds are adjusted dynamically based on the correction value to maintain the robot on the line.

Advantages of Using PID in Line Following

  • Precision: Keeps the robot accurately centered on the line, even in curves.
  • Smooth Movements: Avoids jerky or abrupt adjustments by balancing the contributions of P, I, and D.
  • Disturbance Rejection: Handles changes in surface reflectivity or motor inconsistencies.
  • Adaptability: Can be tuned for different track layouts or robot configurations.

Practical Tuning of PID

  1. Start with Proportional Gain (Kp):

    • Gradually increase Kp until the robot begins to follow the line effectively.
    • Monitor for overshooting or oscillations.
  2. Add Integral Gain (Ki):

    • Introduce Ki to address steady-state errors (e.g., consistent drift).
    • Avoid excessive Ki to prevent instability or oscillations.
  3. Adjust Derivative Gain (Kd):

    • If necessary, use Kd to stabilize movements and reduce oscillations.
    • Avoid setting Kd too high, as it can make the robot sluggish.

Complete Code for TCS34725 Sensor with PID:

Step 5: Programming the Line-Following Robot With TCS34725 RGB Sensor

/**
 * Author: Omar Draidrya
 * Date: 2024/12/06
 * Line-Following Robot with TCS34725.
 */

#include <Wire.h>
#include <AFMotor.h>
#include <Adafruit_TCS34725.h>

// Initialize TCS34725 sensor
Adafruit_TCS34725 tcs = Adafruit_TCS34725(TCS34725_INTEGRATIONTIME_2_4MS, TCS34725_GAIN_4X);

// Motor control pins
#define DIR_A 12
#define DIR_B 13
#define PWM_A 3
#define PWM_B 11

// PID parameters
float Kp = 24.0;  // Proportional gain
float Ki = 1.0;   // Integral gain
float Kd = 1.0;   // Derivative gain
int baseSpeed = 60;  // Base speed

// Middle value for black/white
int middleValue = 3; // Adjusted based on measured values

// Global variables for PID
float error = 0, previousError = 0, sumError = 0;

void setup() {
  Serial.begin(9600);

  // Initialize TCS34725 sensor
  if (!tcs.begin()) {
    Serial.println("Error: TCS34725 not detected. Check wiring!");
    while (1);
  }
  Serial.println("TCS34725 sensor detected!");

  // Configure motor control pins
  pinMode(DIR_A, OUTPUT);
  pinMode(DIR_B, OUTPUT);
  pinMode(PWM_A, OUTPUT);
  pinMode(PWM_B, OUTPUT);

  // Perform sensor calibration
  calibrateSensor();
}

void loop() {
  uint16_t r, g, b, c;
  tcs.getRawData(&r, &g, &b, &c);

  // Calculate error
  error = middleValue - (float)c ;

  // PID calculation
  sumError += error;
  sumError = constrain(sumError, -4, 4); // Limit integral component

  if (abs(error) < 2.0) {
    sumError = 0; // Reset for small errors
  }

  float differential = error - previousError;
  float correction = Kp * error + Ki * sumError + Kd * differential;
  previousError = error;

  // Adjust motor speeds
  int speedLeft = baseSpeed + (int)correction;
  int speedRight = baseSpeed - (int)correction;

  speedLeft = constrain(speedLeft, 20, 255);
  speedRight = constrain(speedRight, 20, 255);

  controlMotors(speedLeft, speedRight);
}

void calibrateSensor() {
  uint16_t r, g, b, c;

  // Measure black value
  Serial.println("Calibrating: Measuring black value...");
  controlMotors(-85, 85); // Slightly turn left
  delay(350);
   // Stop motors after calibration
  controlMotors(0, 0);
  delay(50);

  tcs.getRawData(&r, &g, &b, &c);
  int blackValue = c;
  Serial.print("Black Value: ");
  Serial.println(blackValue);

  // Measure white value
  Serial.println("Calibrating: Measuring white value...");
  controlMotors(85, -85); // Slightly turn right
  delay(500);
    // Stop motors after calibration
  controlMotors(0, 0);
  delay(50);

  tcs.getRawData(&r, &g, &b, &c);
  int whiteValue = c;
  Serial.print("White Value: ");
  Serial.println(whiteValue);

  // Calculate middle value
  middleValue = (blackValue + whiteValue) / 2;
  Serial.print("Middle Value: ");
  Serial.println(middleValue);

  // Stop motors after calibration
  controlMotors(0, 0);
}

void controlMotors(int speedLeft, int speedRight) {
  if (speedLeft >= 0) {
    digitalWrite(DIR_A, HIGH);
  } else {
    digitalWrite(DIR_A, LOW);
    speedLeft = abs(speedLeft);
  }

  if (speedRight >= 0) {
    digitalWrite(DIR_B, HIGH);
  } else {
    digitalWrite(DIR_B, LOW);
    speedRight = abs(speedRight);
  }

  analogWrite(PWM_A, speedLeft);
  analogWrite(PWM_B, speedRight);
}

    

Here’s a detailed breakdown of the code, explaining each section to understand how the TCS34725 sensor-based line-following robot operates:

1. Sensor Initialization

Adafruit_TCS34725 tcs = Adafruit_TCS34725(TCS34725_INTEGRATIONTIME_2_4MS, TCS34725_GAIN_4X);
    
  • The TCS34725 sensor is initialized with a short integration time and moderate gain to optimize light intensity readings for a line-following application.

2. Motor Control Pins

#define DIR_A 12
#define DIR_B 13
#define PWM_A 3
#define PWM_B 11
    
  • DIR_A and DIR_B: Control the direction of the motors.
  • PWM_A and PWM_B: Control the speed of the motors using Pulse Width Modulation (PWM).

3. Calibration Function

void calibrateSensor() {
  // Measures the light intensity for black and white areas.
  // Calculates the middle value (threshold) as the average of the two.
}
    
  • The robot rotates slightly to measure the black (dark) and white (bright) areas.
  • The middleValue is calculated and used as the threshold for distinguishing between the black line and the white surface.

4. Main Loop

void loop() {
  uint16_t r, g, b, c;
  tcs.getRawData(&r, &g, &b, &c);
  
  // Reads the light intensity from the sensor.
  // Calculates error and adjusts motor speeds using PID control.
}
    
  • The light intensity (c) is read from the TCS34725 sensor.
  • PID calculations ensure smooth and accurate adjustments to keep the robot on track.

5. PID Control

float correction = Kp * error + Ki * sumError + Kd * differential;
    
  • Proportional (Kp): Corrects based on the current error.
  • Integral (Ki): Accounts for accumulated error over time.
  • Derivative (Kd): Predicts and responds to changes in error.

6. Motor Control Function

void controlMotors(int speedLeft, int speedRight) {
  // Adjusts motor direction and speed based on the calculated correction.
}
    
  • Controls the direction and speed of each motor to ensure smooth movement along the line.

Step 6: Testing the Line-Following Robot With TCS34725 RGB Sensor

To test the functionality of the robot, a 50mm wide white adhesive tape was placed on a dark surface, forming a curvy line. This setup simulates a typical track for a line-following robot. The testing process was conducted using two different sensor configurations:

  • The Line-Following Robot With TCS34725 RGB Sensor was placed at the left edge of the white tape, with the TCS34725 sensor positioned near the line.
  • The robot performed an automatic calibration by measuring light intensity in two key areas:
    • Dark Area: The robot slightly rotated to measure the dark background.
    • Bright Area: The robot slightly rotated in the opposite direction to measure the white tape.
  • After calculating the middle value as the threshold, the robot began following the line. The PID controller was used to ensure smooth and precise adjustments while navigating curves.

omartronics

Welcome to OmArTronics, the hub for technology enthusiasts and creative minds! I'm Omar, the founder of this website and YouTube channel, and a passionate engineer with a background in electrical and mechanical engineering. I'm currently pursuing my master's in mechatronics in Germany.

Leave a Reply