Description:
This circuit turns the data that the sound sensor gets into an angle based off of a moving average of the amplitude of the data received. The data from the sensor captures the raw sound waves from the air and sends them to the arduino which has a very spiky and unsmooth nature. In addition, the data transitions for increasing to decreasing very rapidly so a traditional filter or tracking the change in the data does not work well at all. Instead, I have implemented a moving average method where the code tracks a certain number of data points, calculates the max and min of the data, and then uses the difference between these values as the new data. This is then multiplied by some multiplier to increase the variance and translated into an angle for a servo to travel to. The algorithm for calculating the angle works wonders while the servo isn’t connected, however, as depicted in the video, the noise of the servo creates an uncontrollable feedback look that results in the servo moving out of control. Regardless, the system responds to noise in the way that it should translating a noise in the environment into an angle for the servo to move to.
Video:
Code:
#include <Servo.h>
//Define Accelerometer Variables for pins
#define MICROPHONE A0
#define SERVO_PIN 9
Servo servo;
double servoAngle = 0;
double alpha = 0.9;
double multiplier = 5;
//Array to hold data for moving average calculation
double speakerData[100]; // {0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
double speakerLevel = 0;
double prevSpeakerLevel = 0;
double maxSpeakerLevel = 200;
double minSpeakerLevel = 50;
void setup() {
//Setup pins as inputs
pinMode(MICROPHONE, INPUT);
//Create servo object
servo.attach(SERVO_PIN);
}
void loop() {
//Update moving average array
for (int i = (sizeof(speakerData)/sizeof(speakerData[0]) - 1); i >= 0; i--) {
if (i != 0) {
speakerData[i] = speakerData[i - 1];
}
else {
speakerData[i] = analogRead(MICROPHONE);
}
}
double MIN = findMin();
double MAX = findMax();
speakerLevel = MAX - MIN;
speakerLevel = multiplier*(alpha*speakerLevel + (1 - alpha)*prevSpeakerLevel); //Filter data
//Put upper and lower bounds on data
if (speakerLevel >= maxSpeakerLevel) {
speakerLevel = maxSpeakerLevel;
}
else if (speakerLevel <= minSpeakerLevel) {
speakerLevel = minSpeakerLevel;
}
servoAngle = 180*(speakerLevel - minSpeakerLevel)/(maxSpeakerLevel - minSpeakerLevel);
servo.write((int) servoAngle);
prevSpeakerLevel = speakerLevel;
}
int findMin() {
double minimum = 0;
for (int i = 0; i < sizeof(speakerData)/sizeof(speakerData[0]); i++) {
if (speakerData[i] < minimum || i == 0) {
minimum = speakerData[i];
}
}
return minimum;
}
int findMax() {
double maximum = 0;
for (int i = 0; i < sizeof(speakerData)/sizeof(speakerData[0]); i++) {
if (speakerData[i] > maximum) {
maximum = speakerData[i];
}
}
return maximum;
}