ME 708 and Friends: Arduino Drawing Robot

by Patrick_Le in Circuits > Arduino

129 Views, 5 Favorites, 0 Comments

ME 708 and Friends: Arduino Drawing Robot

IMG_7653.jpg
Poster.jpg

This project was done for Dr. Wilson's ME 608/708 Mechatronics class at the University of Kansas.

Objective

This project aims to develop a physical system capable of processing a rendering a wide range of images uploaded via a microSD card. The goal is to enable the Arduino Drawing machine to accurately translate digital images into physical drawings with minimal user input

Description

The device draws inspiration from standard FDM 3D printer mechanisms, which can create virtually any object by moving along the X, Y, and Z axes. To develop a drawing robot capable of drawing complex designs, the X and Y mechanisms of a 3D printer were replicated.

To image processing involves first uploading the designated image into MATLAB where the image is processed into a pathway the device can follow. The Arduino that controls the X and Y axis motion of the device is interfaced with MATLAB so that it can receive the pathway.

Supplies

PNG image.png

**The majority of the physical design and all of the 3D printed parts were sourced from the YouTube Channel 'Super Make Something'

Materials

Electronics

1 x Arduino Uno

1 x Breadboard

2 x Nema 17 Stepper Motor

1 x SD Card Breakout Board

2 x A4988 Stepper Motor Driver

1 x Power Plug Connector

30 x Jumper Wires

1 x DC Power Supply

1 x Arduino CNC Shield


3D Printed Parts

2 x Y-Axis Brace

1 x X-Axis Brace

4 x Conduit Bushing

1 x Pen Slider

2 x Motor Drum

2 x X-Axis Carriage Bushing

1 x Carriage Plate

1 x X-Axis Pen Holder Brace

1 x Pen Stop


Hardware

1 x 24"x30" Wooden Base Board

4 x 24"x1-1/4" PVC Pipes

2 x Ball Point Pen Springs

1 x Ball Point Pen

4 x 60mm M3 304 Stainless Steel Dowel Pins

6 x 1/2" #4 Wood Screws

2 x M1.5 Set Screws

113 x M3 Screws

8 x 16mm M4 Screws (with nuts)

20 x 20mm M4 Screws (with nuts)

1 x Roll of String

Construction

Robotic Drawing Machine (Arduino, 3D Printing, CAD, PCB Design, Programming, Stepper Motors)
IMG_7655.jpg
IMG_7657.jpg

The majority of the construction of the device follows along with the video by YouTube channel 'Super Make Something'. Rather than use the custom PCB shown in the video, our device utilizes an Arduino Uno, Arduino CNC shield, and two A4988 motor drivers to drive the two Nema 17 Stepper Motors.

To construct the physical components of the device, follow along with the first YouTube video, starting at 7:34 until 11:51.

Construction Guidelines

  1. Screw the two Motor Drums to the Nema 17 Stepper Motor shafts using the two M1.5 set screws
  2. Attach the Y-Axis Stepper Motor to the Carriage Plate using M3 screws.
  3. Attach two Y-Axis Conduit Bushings to the Carriage Plate using 16mm M4 screws nuts
  4. Attach the remaining Y-Axis Conduit Bushings and the X-Axis bushings to the Carriage Plate using 20mm M4 screws and nuts
  5. Attach the X-Axis Stepper Motor to the X-Axis Conduit Bushing using M3 Screws
  6. Insert the PVC Tubes through the Conduit Bushings (sanding and greasing may be necessary)
  7. Attach the Y-Axis Braces at each end of the Y-Axis PVC Tubes using M4 Screws
  8. Attach the X-Axis Brace onto one end of the X-Axis PVC Tubes using M4 Screws
  9. Press the Dowel Pins into the Pen Stop
  10. Slide the Pen Slider over the Dowel Pins followed by the two Ball Point Pen Springs
  11. Attach the Pen Holder Brace on top of the Dowel Pins
  12. Attach the Pen Holder Assembly onto the other side of the X-Axis PVC Tubes using M4 Screws
  13. Attach the assembly into your wooden base board using the 1/2" #4 Wood Screws
  14. Wrap the string around the Y-Axis motor pulley and feed each end through the Y-Axis Brace
  15. Wrap each end of the spring around an M3 Screw
  16. Repeat the same process for the X-Axis
  17. Insert a pen into the Pen Holder and secure it in place using an M3 Screw

Electronics

IMG_7651.jpg
IMG_7652.jpg
Nema 17 Stepper Motor with Arduino CNC Sheild V3 (A4988 Driver) | 100% Working Arduino Code

To substitute the Arduino Uno in place of the custom PCB used in the video, follow along with the beginning of the second YouTube video by NT Tronix. The video shows how to hook the CNC Shield, A4988 Motor Controller, and Nema 17 Stepper Motors Together.

Construction Guidelines

  1. Place the CNC Shield onto the Arduino Uno making sure the analog and digital pins are lined up
  2. Place the two A4988 Motor Controllers onto the CNC Shield
  3. Attach the two Nema 17 Stepper Motors to the pins directly next to the A4988 Motor Controllers
  4. Attach the SD Card Breakout Board to the CNC Shield using the following pin assignments:
  5. MISO: 12
  6. MOSI: 7
  7. SCK: 13
  8. CS: 4
  9. Set the DC Power Supply to 12V and hook it up to the Arduino Uno

Coding

Arduino Code.png
MATLAB Code.png

Code Walkthrough: Arduino Drawing Robot

This code controls a 2-axis Arduino-powered drawing robot by reading movement instructions from an SD card and driving stepper motors accordingly. Here’s a breakdown of the key components:

1. Libraries and SD Card Initialization

The code uses the SPI (Serial Peripheral Interface) and SD libraries to read movement data from an SD card. The SD card chip select pin is set to pin 10, which is standard for the Arduino Uno. The program attempts to initialize the SD card. If the card or the drawing.txt file cannot be accessed, an error message is displayed in the serial monitor.

2. Pin Configuration

The stepper motors use specific pins for step signals, direction control, microstepping, and enabling/disabling the motors. The X-axis uses pins 2 through 6, while the Y-axis uses pins 7 through A1. The robot uses 1/16 microstepping by setting the microstepping pins to HIGH. This provides smoother, more precise motor movement. The enable pins are set to LOW, activating the stepper motors.

3. Reading Data from the SD Card

The program reads movement data from the drawing.txt file stored on the SD card. The file contains X and Y step values, which determine the robot's movement. The code continuously reads pairs of X and Y step values until the end of the file. If the SD card or file cannot be opened, the program displays an error message and stops execution.

4. Motor Control Logic

The moveStepper() function handles the movement of each motor. It first determines the direction of movement: if the step value is positive, the robot moves forward; if negative, it moves backward. The function then loops through the number of steps, alternating the step pin between HIGH and LOW signals to create the necessary pulses for the stepper motor. The delay between steps is set to 800 microseconds, which increases stability and reduces the chance of missed steps.

5. Main Loop Execution

The main loop continuously reads X and Y step values from the SD card and calls the moveStepper() function to execute the movement. When the robot reaches the end of the file, it closes the file and stops the program by entering an infinite loop.

Key Features and Enhancements

  1. Microstepping Mode: The robot uses 1/16 microstepping, providing finer control and smoother motion.
  2. Smooth and Stable Movement: The step delay is increased from 500 µs to 800 µs for better stability and reliability, preventing missed steps.
  3. Efficient Execution: The robot reads the movement data from the SD card line by line until the file is fully processed, ensuring smooth operation.

How It Works

  1. The robot reads X and Y step values from the drawing.txt file.
  2. It moves the motors accordingly, creating lines or shapes on the drawing surface.
  3. The SD card allows for easy modification of movement patterns by changing the text file.


#include <SPI.h>
#include <SD.h>

File xyValues;
String filename = "drawing.txt";

int csPin = 10; // SD card chip select pin

// X-Axis Stepper Motor Pins
int xStep = 2;
int xDir = 3;
int xMS1 = 4;
int xMS2 = 5;
int xEnable = 6;

// Y-Axis Stepper Motor Pins
int yStep = 7;
int yDir = 8;
int yMS1 = 9;
int yMS2 = A0;
int yEnable = A1;

// Motor Step Delay
int stepDelay = 800;

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

// Initialize stepper pins as outputs
pinMode(xStep, OUTPUT);
pinMode(xDir, OUTPUT);
pinMode(xMS1, OUTPUT);
pinMode(xMS2, OUTPUT);
pinMode(xEnable, OUTPUT);

pinMode(yStep, OUTPUT);
pinMode(yDir, OUTPUT);
pinMode(yMS1, OUTPUT);
pinMode(yMS2, OUTPUT);
pinMode(yEnable, OUTPUT);

// Enable Stepper Drivers (LOW = Enabled)
digitalWrite(xEnable, LOW);
digitalWrite(yEnable, LOW);

// Set microstepping mode to 1/16
digitalWrite(xMS1, HIGH);
digitalWrite(xMS2, HIGH);
digitalWrite(yMS1, HIGH);
digitalWrite(yMS2, HIGH);

// Initialize SD card
if (!SD.begin(csPin)) {
Serial.println("SD card initialization failed!");
return;
}

Serial.println("SD card initialized.");
xyValues = SD.open(filename);
if (!xyValues) {
Serial.println("Error opening drawing.txt");
return;
}
}

void loop() {
if (xyValues.available()) {
int ySteps = xyValues.parseInt(); // Read Y movement
int xSteps = xyValues.parseInt(); // Read X movement

moveStepper(xSteps, xDir, xStep);
moveStepper(ySteps, yDir, yStep);
} else {
xyValues.close();
Serial.println("Finished drawing.");
while (1); // Stop execution
}
}

void moveStepper(int steps, int dirPin, int stepPin) {
if (steps > 0) {
digitalWrite(dirPin, HIGH); // Move forward
} else {
digitalWrite(dirPin, LOW); // Move backward
steps = -steps;
}

for (int i = 0; i < steps; i++) {
digitalWrite(stepPin, HIGH);
delayMicroseconds(stepDelay); // Delay set to 800 µs
digitalWrite(stepPin, LOW);
delayMicroseconds(stepDelay);
}
}


Code Walkthrough: MATLAB Image Halftoning using Floyd-Steinberg Error Diffusion

**Please see the attached code for the credits to the author

This MATLAB program applies Floyd-Steinberg error diffusion to convert a grayscale image into a halftone image. The halftoning process uses dithering to distribute pixel errors, creating the illusion of continuous tones in black-and-white images.

1. Function Definition and Initialization

The program defines a function named floydHalftone() that takes an input grayscale image (inImg) and outputs the halftoned image (outImg).

  1. The input image is converted to double format for precise calculations.
  2. The dimensions M and N represent the number of rows and columns in the image.
  3. The threshold value (T) is set to 127.5, which determines whether a pixel becomes black or white.
  4. An error matrix (y) is created as a copy of the image, which will be modified during error diffusion.

2. Floyd-Steinberg Error Diffusion Algorithm

The algorithm processes the image pixel by pixel, starting from the top-left corner and moving left to right, row by row. The error caused by thresholding is distributed to neighboring pixels according to the Floyd-Steinberg kernel:

  1. 7/16 of the error is passed to the pixel on the right.
  2. 1/16 of the error is passed to the bottom-right pixel.
  3. 5/16 of the error is passed to the pixel directly below.
  4. 3/16 of the error is passed to the bottom-left pixel.

3. Pixel Processing

  1. Left Boundary Handling: The first pixel in each row is processed separately, as it only has neighboring pixels to the right and below.
  2. Center Pixels: The main section of the image processes the central pixels using the full Floyd-Steinberg kernel for error diffusion.
  3. Right Boundary Handling: The last pixel in each row only distributes errors downward and to the bottom-left.
  4. Bottom Row Handling: The final row only diffuses errors horizontally since there are no rows below.

4. Error Distribution Process

For each pixel:

  1. The pixel is thresholded against T.
  2. If the pixel value is greater than or equal to T, it is set to 255 (white).
  3. Otherwise, it is set to 0 (black).
  4. The error between the original and thresholded pixel is calculated.
  5. The error is distributed to the neighboring pixels according to the kernel weights.

5. Final Thresholding and Conversion

After processing all pixels:

  1. The final row undergoes a separate error diffusion process since it has no lower neighbors.
  2. The entire image is thresholded one final time.
  3. The output is converted to binary format using im2bw() to ensure it is displayed as a black-and-white image.

Key Features and Enhancements

  1. Error Diffusion: The algorithm creates a halftone effect by diffusing quantization errors, making the black-and-white image appear smoother and more realistic.
  2. Efficient Pixel Processing: The program handles image boundaries separately to prevent index errors.
  3. Binary Output: The result is a binary image where each pixel is either black or white, making it suitable for printing or display in monochrome formats.

How It Works

  1. The program takes a grayscale image as input.
  2. It processes each pixel, applying the Floyd-Steinberg error diffusion algorithm.
  3. The resulting image appears halftoned, giving the illusion of gray shades using only black and white pixels.


%Program for Image Halftoning by Floyd Method

%Program Description
% The input gray image will be converted into halftone image
% of same size using Floyd's Error Diffusion Method.
%
%Parameters
% inImg - Input Gray Image
% outImg - Output Halftoned Image

%Author : Athi Narayanan S
%Student, M.E, EST,
%K.S.R College of Engineering
%Erode, Tamil Nadu, India.
%s_athi1983@yahoo.co.in
%http://sites.google.com/site/athisnarayanan/

function outImg = floydHalftone(inImg)
inImg = double(inImg);

[M,N] = size(inImg);
T = 127.5; %Threshold
y = inImg;
error = 0;

for rows = 1:M-1

%Left Boundary of Image
outImg(rows,1) =255*(y(rows,1)>=T);
error = -outImg(rows,1) + y(rows,1);
y(rows,1+1) = 7/16 * error + y(rows,1+1);
y(rows+1,1+1) = 1/16 * error + y(rows+1,1+1);
y(rows+1,1) = 5/16 * error + y(rows+1,1);

for cols = 2:N-1
%Center of Image
outImg(rows,cols) =255*(y(rows,cols)>=T);
error = -outImg(rows,cols) + y(rows,cols);
y(rows,cols+1) = 7/16 * error + y(rows,cols+1);
y(rows+1,cols+1) = 1/16 * error + y(rows+1,cols+1);
y(rows+1,cols) = 5/16 * error + y(rows+1,cols);
y(rows+1,cols-1) = 3/16 * error + y(rows+1,cols-1);
end

%Right Boundary of Image
outImg(rows,N) =255*(y(rows,N)>=T);
error = -outImg(rows,N) + y(rows,N);
y(rows+1,N) = 5/16 * error + y(rows+1,N);
y(rows+1,N-1) = 3/16 * error + y(rows+1,N-1);

end

%Bottom & Left Boundary of Image
rows = M;
outImg(rows,1) =255*(y(rows,1)>=T);
error = -outImg(rows,1) + y(rows,1);
y(rows,1+1) = 7/16 * error + y(rows,1+1);

%Bottom & Center of Image
for cols = 2:N-1
outImg(rows,cols) =255*(y(rows,cols)>=T);
error = -outImg(rows,cols) + y(rows,cols);
y(rows,cols+1) = 7/16 * error + y(rows,cols+1);
end

%Thresholding
outImg(rows,N) =255*(y(rows,N)>=T);

outImg = im2bw(uint8(outImg));

Demonstration

ME708

Please enjoy a brief demonstration video of our project in action!