| name | moving-rainbow |
| description | Generate MicroPython programs for the Moving Rainbow LED strip educational project using Raspberry Pi Pico with NeoPixel strips and button controls. |
Moving Rainbow MicroPython Program Generator
This skill helps create MicroPython programs for the Moving Rainbow educational project. Use this skill when the user asks to create LED animations, NeoPixel programs, or Moving Rainbow examples for Raspberry Pi Pico.
Hardware Configuration
The default hardware setup consists of:
- Microcontroller: Raspberry Pi Pico (RP2040)
- LED Strip: 30-pixel NeoPixel/WS2812B addressable LED strip connected to GPIO pin 0
- Input Controls: Two momentary push buttons
- Button 1: GPIO pin 14
- Button 2: GPIO pin 15
- Built-in LED: GPIO pin 25 (can be used for status indication)
Configuration values are stored in a config.py file that should be imported by programs. See the references/config.py file for the standard configuration template.
Code Structure and Patterns
Basic Program Template
All programs should follow this structure:
from machine import Pin
from neopixel import NeoPixel
from utime import sleep
import config
# Initialize the LED strip
strip = NeoPixel(Pin(config.NEOPIXEL_PIN), config.NUMBER_PIXELS)
# Your code here
while True:
# Animation loop
pass
Essential Components
- Imports: Always import from
machine,neopixel,utime, and theconfigmodule - Strip Initialization: Create the NeoPixel object using configuration values
- Main Loop: Use a
while True:loop for continuous animations - Color Format: Colors are RGB tuples like
(red, green, blue)with values 0-255
Common Functions and Patterns
Color Wheel Function
For smooth rainbow transitions, use the standard color wheel function:
def wheel(pos):
# Input a value 0 to 255 to get a color value.
# The colors are a transition r - g - b - back to r.
if pos < 0 or pos > 255:
return (0, 0, 0)
if pos < 85:
return (255 - pos * 3, pos * 3, 0)
if pos < 170:
pos -= 85
return (0, 255 - pos * 3, pos * 3)
pos -= 170
return (pos * 3, 0, 255 - pos * 3)
Strip Control Patterns
Setting pixels:
strip[index] = (red_value, green_value, blue_value)
strip.write() # Always call write() to display changes
Erasing the strip:
def erase():
for i in range(0, config.NUMBER_PIXELS):
strip[i] = (0, 0, 0)
strip.write()
Using a counter with modulo for wrapping:
counter = 0
while True:
# Use counter for position
strip[counter] = color
strip.write()
sleep(delay)
counter += 1
counter = counter % config.NUMBER_PIXELS # Wrap around
Button Integration
For interactive programs with button controls:
from machine import Pin
from utime import ticks_ms
import config
BUTTON_PIN_1 = config.BUTTON_PIN_1
BUTTON_PIN_2 = config.BUTTON_PIN_2
button1 = Pin(BUTTON_PIN_1, Pin.IN, Pin.PULL_DOWN)
button2 = Pin(BUTTON_PIN_2, Pin.IN, Pin.PULL_DOWN)
last_time = 0
def button_pressed_handler(pin):
global mode, last_time
new_time = ticks_ms()
# Debounce: require 200ms between button presses
if (new_time - last_time) > 200:
pin_num = int(str(pin)[4:6])
if pin_num == BUTTON_PIN_1:
# Button 1 action (e.g., increment mode)
mode += 1
else:
# Button 2 action (e.g., decrement mode)
mode -= 1
last_time = new_time
# Register interrupt handlers
button1.irq(trigger=Pin.IRQ_FALLING, handler=button_pressed_handler)
button2.irq(trigger=Pin.IRQ_FALLING, handler=button_pressed_handler)
Common Animation Patterns
Moving Dot
def move_dot(counter, color, delay):
strip[counter] = color
strip.write()
sleep(delay)
strip[counter] = (0, 0, 0)
Color Wipe
def color_wipe(color, delay):
for i in range(config.NUMBER_PIXELS):
strip[i] = color
strip.write()
sleep(delay)
Rainbow Cycle
def rainbow_cycle(counter, delay):
percent_color_wheel = round(255 / config.NUMBER_PIXELS)
for i in range(0, config.NUMBER_PIXELS):
color_index = round(i * percent_color_wheel)
color = wheel(color_index)
strip[(i + counter) % config.NUMBER_PIXELS] = color
strip.write()
sleep(delay)
Comet Tail Effect
def comet_tail(counter, color, tail_length, delay):
levels = [255, 128, 64, 32, 16, 8, 4, 2, 1]
for i in range(0, tail_length):
target = (counter - i) % config.NUMBER_PIXELS
scale = levels[i] / 255
strip[target] = (int(color[0]*scale), int(color[1]*scale), int(color[2]*scale))
strip.write()
sleep(delay)
Random Effects
from urandom import randint
def random_color(delay):
random_offset = randint(0, config.NUMBER_PIXELS - 1)
random_color_value = randint(0, 255)
strip[random_offset] = wheel(random_color_value)
strip.write()
sleep(delay)
Multi-Mode Programs
For programs with multiple animation modes that can be switched with buttons:
mode_list = ['moving rainbow', 'red dot', 'blue dot', 'candle flicker', 'random']
mode_count = len(mode_list)
mode = 0
last_mode = -1
while True:
# Print mode changes
if mode != last_mode:
print('mode=', mode, 'running', mode_list[mode])
last_mode = mode
# Execute the current mode
if mode == 0:
moving_rainbow(counter, 0.05)
elif mode == 1:
move_dot(counter, red, 0.05)
# ... more modes
counter += 1
counter = counter % config.NUMBER_PIXELS
Standard Color Definitions
Use these common color constants:
red = (255, 0, 0)
orange = (255, 60, 0)
yellow = (255, 150, 0)
green = (0, 255, 0)
blue = (0, 0, 255)
cyan = (0, 255, 255)
indigo = (75, 0, 130)
violet = (138, 43, 226)
white = (128, 128, 128)
off = (0, 0, 0)
Educational Principles
When generating programs, follow these educational guidelines:
- Progressive Complexity: Start simple and add features incrementally
- Clear Comments: Explain what each section does for learning purposes
- Consistent Naming: Use descriptive variable names (e.g.,
counter,delay,color) - Visible Feedback: Use print statements to show what's happening
- Adjustable Parameters: Use constants for delays and other values so students can experiment
Best Practices
- Always call
strip.write()after modifying pixels to display changes - Use modulo for wrapping:
counter % config.NUMBER_PIXELSto loop animations - Debounce buttons: Check that at least 200ms has passed between button presses
- Import config: Always use
import configand referenceconfig.NEOPIXEL_PIN, etc. - Add delays: Include appropriate
sleep()calls to control animation speed - Clear pixels: Turn off pixels when moving animations to prevent trails
- Test boundary conditions: Ensure animations work correctly at pixel 0 and the last pixel
When to Use This Skill
Use this skill when:
- Creating LED animation programs for Raspberry Pi Pico
- Working with NeoPixel/WS2812B addressable LED strips
- Building educational examples for the Moving Rainbow project
- Implementing button-controlled LED effects
- Generating MicroPython code for LED strip projects
Output Format
Generated programs should:
- Be complete, runnable MicroPython code
- Include necessary imports
- Use the config module for hardware settings
- Include helpful comments
- Follow the established code patterns
- Be educational and easy to understand