Basics of Potentiometers with Arduino
Learn the fundamentals of how a potentiometers works, about the forms they come in, and how to use them in your projects.
A potentiometer is a simple mechanical device that comes in many different forms. It provides a variable amount of resistance that changes as you manipulate it. The examples in this article uses a potentiometer with a twisting shaft, one of the more common versions of a potentiometer you will find.
By passing voltage through a potentiometer into an analog input on your Arduino, it is possible to measure the amount of resistance of the potentiometer as an analog value. This article will showcase use cases of potentiometers, as well as teach you how to connect and read data from them. One shows how you can use a potentiometer as an input for a color mixer, and the other shows how to accurately choose colors and how to smoothly fade between them.
The typical potentiometer will have 3 pins, two power supply pins (+5V and GND), and one pin that connects to an analog input pin on your Arduino to read the value output.
To learn how to read data from a potentiometer, and display it in the Serial Monitor, visit the Analog Read Serial example.
Hardware Required
- Arduino board
- Potentiometer
- 1x Red, 1x Green, 1x Blue LED
- 3x 220 Ohm Resistors
Circuit
- Potentiometer + Pin to 5V
- Potentiometer - Pin to GND
- Potentiometer Data Pin to A3
- A Red LED connected to pin 9 with a 220 Ohm Resistor
- A Green LED connected to pin 10 with a 220 Ohm Resistor
- A Blue LED connected to pin 11 with a 220 Ohm Resistor
Color Mixer Example
This example will show you how a potentiometer can be used as an analog input to mix colors with great granularity.
1/*2* Code for making one potentiometer control 3 LEDs, red, grn and blu, or one tri-color LED3* The program cross-fades from red to grn, grn to blu, and blu to red4* Clay Shirky <clay.shirky@nyu.edu> 5*/6
7// INPUT: Potentiometer should be connected to 5V and GND8int potPin = A3; // Potentiometer output connected to analog pin 39int potVal = 0; // Variable to store the input from the potentiometer10
11// OUTPUT: Use digital pins 9-11, the Pulse-width Modulation (PWM) pins12// LED's cathodes should be connected to digital GND13int redPin = 9; // Red LED, connected to digital pin 914int grnPin = 10; // Green LED, connected to digital pin 1015int bluPin = 11; // Blue LED, connected to digital pin 1116
17// Program variables18int redVal = 0; // Variables to store the values to send to the pins19int grnVal = 0;20int bluVal = 0;21
22void setup()23{24 pinMode(redPin, OUTPUT); // sets the pins as output25 pinMode(grnPin, OUTPUT); 26 pinMode(bluPin, OUTPUT); 27}28
29// Main program30void loop()31{32 potVal = analogRead(potPin); // read the potentiometer value at the input pin33
34 if (potVal < 341) // Lowest third of the potentiometer's range (0-340)35 { 36 potVal = (potVal * 3) / 4; // Normalize to 0-25537
38 redVal = 256 - potVal; // Red from full to off39 grnVal = potVal; // Green from off to full40 bluVal = 1; // Blue off41 }42 else if (potVal < 682) // Middle third of potentiometer's range (341-681)43 {44 potVal = ( (potVal-341) * 3) / 4; // Normalize to 0-25545
46 redVal = 1; // Red off47 grnVal = 256 - potVal; // Green from full to off48 bluVal = potVal; // Blue from off to full49 }50 else // Upper third of potentiometer"s range (682-1023)51 {52 potVal = ( (potVal-683) * 3) / 4; // Normalize to 0-25553
54 redVal = potVal; // Red from off to full55 grnVal = 1; // Green off56 bluVal = 256 - potVal; // Blue from full to off57 }58 analogWrite(redPin, redVal); // Write values to LED pins59 analogWrite(grnPin, grnVal); 60 analogWrite(bluPin, bluVal); 61}
Color Crossfading Example
1/*2* Code for cross-fading 3 LEDs, red, green and blue (RGB) 3* To create fades, you need to do two things: 4* 1. Describe the colors you want to be displayed5* 2. List the order you want them to fade in6*7* DESCRIBING A COLOR:8* A color is just an array of three percentages, 0-100, 9* controlling the red, green and blue LEDs10*11* Red is the red LED at full, blue and green off12* int red = { 100, 0, 0 }13* Dim white is all three LEDs at 30%14* int dimWhite = {30, 30, 30}15* etc.16*17* Some common colors are provided below, or make your own18* 19* LISTING THE ORDER:20* In the main part of the program, you need to list the order 21* you want to colors to appear in, e.g.22* crossFade(red);23* crossFade(green);24* crossFade(blue);25*26* Those colors will appear in that order, fading out of 27* one color and into the next 28*29* In addition, there are 5 optional settings you can adjust:30* 1. The initial color is set to black (so the first color fades in), but 31* you can set the initial color to be any other color32* 2. The internal loop runs for 1020 iterations; the 'wait' variable33* sets the approximate duration of a single crossfade. In theory, 34* a 'wait' of 10 ms should make a crossFade of ~10 seconds. In 35* practice, the other functions the code is performing slow this 36* down to ~11 seconds on my board. YMMV.37* 3. If 'repeat' is set to 0, the program will loop indefinitely.38* if it is set to a number, it will loop that number of times,39* then stop on the last color in the sequence. (Set 'return' to 1, 40* and make the last color black if you want it to fade out at the end.)41* 4. There is an optional 'hold' variable, which pasues the 42* program for 'hold' milliseconds when a color is complete, 43* but before the next color starts.44* 5. Set the DEBUG flag to 1 if you want debugging output to be45* sent to the serial monitor.46*47* The internals of the program aren't complicated, but they48* are a little fussy -- the inner workings are explained 49* below the main loop.50*51* April 2007, Clay Shirky <clay.shirky@nyu.edu> 52*/ 53
54// Output55int redPin = 9; // Red LED, connected to digital pin 956int grnPin = 10; // Green LED, connected to digital pin 1057int bluPin = 11; // Blue LED, connected to digital pin 1158
59// Color arrays60int black[3] = { 0, 0, 0 };61int white[3] = { 100, 100, 100 };62int red[3] = { 100, 0, 0 };63int green[3] = { 0, 100, 0 };64int blue[3] = { 0, 0, 100 };65int yellow[3] = { 40, 95, 0 };66int dimWhite[3] = { 30, 30, 30 };67// etc.68
69// Set initial color70int redVal = black[0];71int grnVal = black[1]; 72int bluVal = black[2];73
74int wait = 10; // 10ms internal crossFade delay; increase for slower fades75int hold = 0; // Optional hold when a color is complete, before the next crossFade76int DEBUG = 1; // DEBUG counter; if set to 1, will write values back via serial77int loopCount = 60; // How often should DEBUG report?78int repeat = 3; // How many times should we loop before stopping? (0 for no stop)79int j = 0; // Loop counter for repeat80
81// Initialize color variables82int prevR = redVal;83int prevG = grnVal;84int prevB = bluVal;85
86// Set up the LED outputs87void setup()88{89 pinMode(redPin, OUTPUT); // sets the pins as output90 pinMode(grnPin, OUTPUT); 91 pinMode(bluPin, OUTPUT); 92
93 if (DEBUG) { // If we want to see values for debugging...94 Serial.begin(9600); // ...set up the serial output 95 }96}97
98// Main program: list the order of crossfades99void loop()100{101 crossFade(red);102 crossFade(green);103 crossFade(blue);104 crossFade(yellow);105
106 if (repeat) { // Do we loop a finite number of times?107 j += 1;108 if (j >= repeat) { // Are we there yet?109 exit(j); // If so, stop.110 }111 }112}113
114/* BELOW THIS LINE IS THE MATH -- YOU SHOULDN'T NEED TO CHANGE THIS FOR THE BASICS115* 116* The program works like this:117* Imagine a crossfade that moves the red LED from 0-10, 118* the green from 0-5, and the blue from 10 to 7, in119* ten steps.120* We'd want to count the 10 steps and increase or 121* decrease color values in evenly stepped increments.122* Imagine a + indicates raising a value by 1, and a -123* equals lowering it. Our 10 step fade would look like:124* 125* 1 2 3 4 5 6 7 8 9 10126* R + + + + + + + + + +127* G + + + + +128* B - - -129* 130* The red rises from 0 to 10 in ten steps, the green from 131* 0-5 in 5 steps, and the blue falls from 10 to 7 in three steps.132* 133* In the real program, the color percentages are converted to 134* 0-255 values, and there are 1020 steps (255*4).135* 136* To figure out how big a step there should be between one up- or137* down-tick of one of the LED values, we call calculateStep(), 138* which calculates the absolute gap between the start and end values, 139* and then divides that gap by 1020 to determine the size of the step 140* between adjustments in the value.141*/142
143int calculateStep(int prevValue, int endValue) {144 int step = endValue - prevValue; // What's the overall gap?145 if (step) { // If its non-zero, 146 step = 1020/step; // divide by 1020147 } 148 return step;149}150
151/* The next function is calculateVal. When the loop value, i,152* reaches the step size appropriate for one of the153* colors, it increases or decreases the value of that color by 1. 154* (R, G, and B are each calculated separately.)155*/156
157int calculateVal(int step, int val, int i) {158
159 if ((step) && i % step == 0) { // If step is non-zero and its time to change a value,160 if (step > 0) { // increment the value if step is positive...161 val += 1; 162 } 163 else if (step < 0) { // ...or decrement it if step is negative164 val -= 1;165 } 166 }167 // Defensive driving: make sure val stays in the range 0-255168 if (val > 255) {169 val = 255;170 } 171 else if (val < 0) {172 val = 0;173 }174 return val;175}176
177/* crossFade() converts the percentage colors to a 178* 0-255 range, then loops 1020 times, checking to see if 179* the value needs to be updated each time, then writing180* the color values to the correct pins.181*/182
183void crossFade(int color[3]) {184 // Convert to 0-255185 int R = (color[0] * 255) / 100;186 int G = (color[1] * 255) / 100;187 int B = (color[2] * 255) / 100;188
189 int stepR = calculateStep(prevR, R);190 int stepG = calculateStep(prevG, G); 191 int stepB = calculateStep(prevB, B);192
193 for (int i = 0; i <= 1020; i++) {194 redVal = calculateVal(stepR, redVal, i);195 grnVal = calculateVal(stepG, grnVal, i);196 bluVal = calculateVal(stepB, bluVal, i);197
198 analogWrite(redPin, redVal); // Write current values to LED pins199 analogWrite(grnPin, grnVal); 200 analogWrite(bluPin, bluVal); 201
202 delay(wait); // Pause for 'wait' milliseconds before resuming the loop203
204 if (DEBUG) { // If we want serial output, print it at the 205 if (i == 0 or i % loopCount == 0) { // beginning, and every loopCount times206 Serial.print("Loop/RGB: #");207 Serial.print(i);208 Serial.print(" | ");209 Serial.print(redVal);210 Serial.print(" / ");211 Serial.print(grnVal);212 Serial.print(" / "); 213 Serial.println(bluVal); 214 } 215 DEBUG += 1;216 }217 }218 // Update current values for next loop219 prevR = redVal; 220 prevG = grnVal; 221 prevB = bluVal;222 delay(hold); // Pause for optional 'wait' milliseconds before resuming the loop223}
Suggest changes
The content on docs.arduino.cc is facilitated through a public GitHub repository. If you see anything wrong, you can edit this page here.
License
The Arduino documentation is licensed under the Creative Commons Attribution-Share Alike 4.0 license.