Taula de continguts:
2025 Autora: John Day | [email protected]. Última modificació: 2025-01-13 06:57
En aquest tutorial jugarem amb un sensor de gestos (APDS-9960) i un anell de neopixel per aprendre a combinar-los amb un Arduino UNO.
El producte final respondrà als gestos esquerra-dreta animant el moviment del led cap a la dreta o cap a l’esquerra, i als gestos cap amunt i cap avall canviant el color dels leds.
En els passos següents, veureu breument la llista de peces i com connectar els components. A continuació, revisarem el codi pas a pas per saber com funciona.
Pas 1: components
1. Arduino UNO
2. cable USB
3. Sensor de gestos APDS9960 (https://www.sparkfun.com/products/12787)
4. Anell de neopixels de 24 leds (https://www.adafruit.com/product/1586)
5. Cables de taulers de connexió mascle-femella, mascle-mascle
6. taulell de pa
7. Font d'alimentació de 5 V per a l'anell LED (estic fent servir una bateria de 4)
8. Per fixar l'anell de neopíxels a la placa de marc, haureu de soldar-hi tres pins masculins: GND, PWR i pin de control. Per a això, necessitareu un soldador i un flux
Els components principals són el sensor de gest APDS-9960 i l’anell de 24 neopíxels. Podeu canviar diferents arduinos, fonts d'alimentació de cables USB i taulers de configuració com vulgueu.
Pas 2: Muntatge i càrrega
muntatge
Abans de començar, assegureu-vos que teniu tots els components a la taula. Tindrem alguns bons passos a seguir:). També he adjuntat l’esquema de Fritzing com a imatge i també en format fritzing.
1. Soldeu 3 pins masculins a l'anell de neopixel (GND, PWR, pin de control)
2. connecteu l'anell de neopixel a la taula de tall
3. connecteu el sensor APDS9960 a la placa de control
4. connecteu els terrenys: bateria, arduino UNO, APDS9960 i neopixel a la placa de terra
5. connecteu l'alimentació: arduino UNO 3V al pin d'alimentació APDS9960, neopixel a la bateria
6. connecteu el pin de control del neopixel al pin arduino D6
7. connecteu SDA i SCL de l'APDS9960 a l'A4 i A5 respectivament
8. connecteu el pin d'interrupció APDS9960 a l'arduino D2
Càrrega de codi
Primer de tot, haureu de descarregar i instal·lar les biblioteques arduino necessàries:
1. Biblioteca d'anells Neopixel:
2. Biblioteca del sensor de gestos:
Si no sabeu com instal·lar biblioteques arduino, consulteu aquest tutorial.
Després de descarregar i instal·lar les biblioteques anteriors, podeu clonar o descarregar el meu dipòsit arduino situat aquí: https://github.com/danionescu0/arduino i utilitzarem aquest esbós: https://github.com/danionescu0 / arduino / tree / master / projects / neopixel_ring_gestures
A la següent secció inseriré el codi directament en aquest tutorial, de manera que si voleu el podeu copiar i enganxar des d'allà.
Finalment, connecteu l’arduino a l’ordinador mitjançant el cable USB, poseu piles d’1,5 V a la bateria i pengeu l’esbós a l’arduino.
Pas 3: Com funciona?
En aquesta última part, aprendrem com es combinen aquests components, com utilitzar les seves biblioteques i com he estructurat el meu codi:
Primer fem una ullada ràpida a través del mètode de l'API de la biblioteca de neopíxels i del sensor que utilitzarem
1. API Neopixel d'Adafruit
Des d’aquesta biblioteca farem servir els mètodes que controlen i apliquen el color del led individual
- inclou la biblioteca:
#incloure
- declarar la biblioteca
#define NEOPIXED_CONTROL_PIN 6
#define NUM_LEDS 24 Adafruit_NeoPixel strip = Adafruit_NeoPixel (NUM_LEDS, NEOPIXED_CONTROL_PIN, NEO_RBG + NEO_KHZ800);
- inicialitzar
#típicament dins del bloc de configuració
void setup () {strip.begin (); # potser alguna cosa més aquí # … }
- il·lumineu píxels individuals i apliqueu totes les modificacions a la tira (renderitzeu-la d'una manera)
# configureu el píxel 0 perquè sigui vermell
strip.setPixelColor (0, strip. Color (255, 0, 0)); # configureu el píxel 1 perquè sigui verd strip.setPixelColor (1, strip. Color (0, 255, 0)); # configureu el píxel 2 perquè sigui blau strip.setPixelColor (2, strip. Color (0, 0 255)); strip.show ();
2. Sensor de gestos APDS 9960
Des d'aquesta biblioteca farem servir la funció "gest de lectura". Aquesta funció serà capaç de distingir entre ordres esquerra-dreta, amunt-avall i molt a prop. Aquí hi ha un truc, no anem a demanar al sensor contínuament l’últim gest percebut. La junta té la capacitat de "fer ping" mitjançant una interrupció que ha trobat un gest.
- inclou la biblioteca, similar al neopixel
- declarar la biblioteca el pin d'interrupció i el senyal d'interrupció
#define APDS9960_INT 2
SparkFun_APDS9960 apds = SparkFun_APDS9960 (); int isr_flag = 0;
- inicialitzar la biblioteca, normalment dins de la funció de configuració
configuració nul·la ()
{# declarar el pin d'interrupció com a INPUT i adjuntar-hi una funció pinMode (APDS9960_INT, INPUT); attachInterrupt (0, interruptRoutine, FALLING); if (apds.init () && apds.enableGestureSensor (true)) {Serial.println ("S'ha completat la inicialització APDS-9960"); } else {Serial.println ("S'ha produït un error durant l'inici de l'APDS-9960!"); } # inicialitzeu altres coses potser}
- definiu la funció d'interrupció, aquí establirem només una marca
void interruptRoutine () {
isr_flag = 1; }
- dins de la funció de bucle comproveu la senyal periòdicament per veure si s’ha detectat un gest
bucle buit ()
{# comproveu la marca si (isr_flag == 1) {# si la marca està configurada, elimineu la interrupció, feu el processament necessari a la funció handleGesture () # i, a continuació, reinicieu la marca i torneu a connectar la interrupció detachInterrupt (0); handleGesture (); isr_flag = 0; attachInterrupt (0, interruptRoutine, FALLING); } # algun altre codi aquí potser}
- defineix la funció handleGesture () on podem demanar l'últim gest
void handleGesture () {
# si no hi ha cap gest disponible, només és una comprovació segura si (! apds.isGestureAvailable ()) {return; } # llegeix l'últim gest, es compara amb els coneguts i imprimeix un commutador de missatges (apds.readGesture ()) {case DIR_UP: Serial.println ("UP"); trencar; cas DIR_DOWN: Serial.println ("DOWN"); trencar; cas DIR_LEFT: Serial.println ("ESQUERRA"); trencar; cas DIR_RIGHT: Serial.println ("DRETA"); trencar; cas DIR_FAR: Serial.println ("FAR"); trencar; }}
Ara vegem tot el codi en acció:
Així que he explicat l’API base del sensor de gestos i de l’anell de neopíxels, ara juntem les coses:
L'algorisme funciona així:
- inicialitzar les biblioteques (vegeu el codi anterior)
- Creeu un conjunt d'intensitats de leds anomenades "ledStates". Aquesta matriu contindrà 24 intensitats de leds que es disposen de manera descendent de 150 a 2
- dins del bucle principal comproveu si s'ha modificat el pin d'interrupció, si és així, és hora de canviar l'animació o el color del led
- la funció "handleGesture ()" comprova l'últim gest i crida a la funció "toggleColor" per als gestos AMUNT-BAIX o estableix una variable global "ledDirection" per als gestos ESQUERRA - DRETA
- la funció "toggleColor ()" simplement canvia una variable global anomenada "colorSelection" amb un dels valors 0, 1, 2
- també dins de la funció de bucle principal una altra funció anomenada "animateLeds ();" es diu. Aquesta funció comprova si passen 100 mil·lisegons i, en aquest cas, gira els leds mitjançant la funció "rotateLeds ()" i els torna a dibuixar
- el "rotateLeds ()" farà "rotar" els leds cap endavant o cap enrere mitjançant l'ús d'una altra matriu anomenada "intermediateLedStates".
L'efecte de rotació serà així:
# després de la inicialització
{150, 100, 70, 50, 40, 30, 10, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; # després de rotateLeds () s'anomena {0, 150, 100, 70, 50, 40, 30, 10, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; # després de rotateLeds () es torna a cridar {0, 0, 150, 100, 70, 50, 40, 30, 10, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; # etcètera
Per a això, primer es crea la nova matriu i es copia les antigues intensitats de leds a les noves posicions (incrementar la posició o disminuir-la). Després, sobreescriu la matriu "ledStates" amb el "intermediateLedStates", de manera que el procés continuarà després de 100 mil·lisegons més.
#include "SparkFun_APDS9960.h"
#include "Adafruit_NeoPixel.h"
#include "Wire.h" #define NEOPIXED_CONTROL_PIN 6 #define NUM_LEDS 24 #define APDS9960_INT 2 #define LED_SPEED_STEP_INTERVAL 100 Adafruit_NeoPixel strip = Adafruit_NeoPixel (NUM_LEDS, NEOPIXED_CONTROL_P; SparkFun_APDS9960 apds = SparkFun_APDS9960 (); unsigned long lastLedChangeTime = 0; direcció LED curta = 0; selecció de color curta = 0; byte ledStates = {150, 100, 70, 50, 40, 30, 10, 2, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0}; int isr_flag = 0; configuració nul·la () {Serial.begin (9600); Serial.println ("S'ha iniciat el programa"); strip.begin (); pinMode (APDS9960_INT, INPUT); attachInterrupt (0, interruptRoutine, FALLING); if (apds.init () && apds.enableGestureSensor (true)) {Serial.println ("S'ha completat la inicialització APDS-9960"); } else {Serial.println ("S'ha produït un error durant l'inici de l'APDS-9960!"); } lastLedChangeTime = millis (); Serial.println ("Inici amb èxit"); } void loop () {if (isr_flag == 1) {detachInterrupt (0); handleGesture (); isr_flag = 0; attachInterrupt (0, interruptRoutine, FALLING); } animateLeds (); } void interruptRoutine () {isr_flag = 1; } / ** * Això gestionarà els gestos del sensor APDS9960 * Els gestos amunt i avall cridaran a la funció toggleColor * Els gestos esquerre i dret canviaran l'animació del led * / void handleGesture () {if (! Apds.isGestureAvailable ()) {return; } switch (apds.readGesture ()) {case DIR_UP: Serial.println ("UP"); toggleColor (); trencar; cas DIR_DOWN: Serial.println ("BAIX"); toggleColor (); trencar; cas DIR_LEFT: ledDirection = 1; Serial.println ("ESQUERRA"); trencar; cas DIR_RIGHT: ledDirection = -1; Serial.println ("DRETA"); trencar; cas DIR_FAR: ledDirection = 0; Serial.println ("FAR"); trencar; }} / ** * Canvia el color dels leds actuals * Cada vegada que es crida aquesta funció canviarà l'estat dels leds * / void toggleColor () {if (colorSelection == 0) {colorSelection = 1; } else if (colorSelection == 1) {colorSelection = 2; } else {colorSelection = 0; }} / ** * L'animació s'executarà després de LED_SPEED_STEP_INTERVAL millis * Primer es crida la funció rotateLeds, després es configuren els colors dels leds mitjançant la tira api * / void animateLeds () {if (millis () - lastLedChangeTime <LED_SPEED_STEP_INTERVAL) {return; } rotateLeds (); for (int i = 0; i <NUM_LEDS; i ++) {strip.setPixelColor (i, getColor (ledStates )); strip.show (); } lastLedChangeTime = millis (); } / ** * Utilitzant una matriu secundària "intermediateLedStates", s'intenten intensitats de leds * Primer es copien els valors de "ledStates" a "intermediateLedStates" com ara * sentem-nos que la matriu "ledStates" és {100, 80, 60, 0, 0, 0} i ledDirection és 1 *, després que aquesta funció s'anomeni "ledStates", la matriu és {0, 100, 80, 60, 0, 0} simulant un efecte de rotació * / void rotateLeds () {byte intermediateLedStates [NUM_LEDS]; for (int i = 0; i <NUM_LEDS; i ++) {intermèdiaStats = 0; } for (int i = 0; i <NUM_LEDS; i ++) {if (ledDirection == 1) {if (i == NUM_LEDS -1) {intermediateLedStates [0] = ledStates ; } else {intermitentsEstats [i + 1] = estats led ; }} else {if (i == 0) {intermediateLedStates [NUM_LEDS - 1] = ledStates ; } else {intermitentsEstats [i - 1] = ledEstats ; }}} per a (int i = 0; i <NUM_LEDS; i ++) {ledStates = intermediarisLedStates ; }} uint32_t getColor (int intensitat) {commutador (colorSelecció) {cas 0: tira de retorn. Color (intensitat, 0, 0); cas 1: tira de retorn. Color (0, intensitat, 0); per defecte: return strip. Color (0, 0, intensitat); }}
Espero que us hagi agradat, podeu fer servir la secció de comentaris per fer-me preguntes.