home / skills / vishalsachdev / claude-skills / moving-rainbow
This skill helps you generate micropython animations for the Moving Rainbow project on Raspberry Pi Pico using NeoPixel strips and button controls.
npx playbooks add skill vishalsachdev/claude-skills --skill moving-rainbowReview the files below or copy the command above to add this skill to your agents.
---
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:
```python
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
1. **Imports**: Always import from `machine`, `neopixel`, `utime`, and the `config` module
2. **Strip Initialization**: Create the NeoPixel object using configuration values
3. **Main Loop**: Use a `while True:` loop for continuous animations
4. **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:
```python
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**:
```python
strip[index] = (red_value, green_value, blue_value)
strip.write() # Always call write() to display changes
```
**Erasing the strip**:
```python
def erase():
for i in range(0, config.NUMBER_PIXELS):
strip[i] = (0, 0, 0)
strip.write()
```
**Using a counter with modulo for wrapping**:
```python
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:
```python
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
```python
def move_dot(counter, color, delay):
strip[counter] = color
strip.write()
sleep(delay)
strip[counter] = (0, 0, 0)
```
#### Color Wipe
```python
def color_wipe(color, delay):
for i in range(config.NUMBER_PIXELS):
strip[i] = color
strip.write()
sleep(delay)
```
#### Rainbow Cycle
```python
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
```python
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
```python
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:
```python
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:
```python
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:
1. **Progressive Complexity**: Start simple and add features incrementally
2. **Clear Comments**: Explain what each section does for learning purposes
3. **Consistent Naming**: Use descriptive variable names (e.g., `counter`, `delay`, `color`)
4. **Visible Feedback**: Use print statements to show what's happening
5. **Adjustable Parameters**: Use constants for delays and other values so students can experiment
## Best Practices
1. **Always call `strip.write()`** after modifying pixels to display changes
2. **Use modulo for wrapping**: `counter % config.NUMBER_PIXELS` to loop animations
3. **Debounce buttons**: Check that at least 200ms has passed between button presses
4. **Import config**: Always use `import config` and reference `config.NEOPIXEL_PIN`, etc.
5. **Add delays**: Include appropriate `sleep()` calls to control animation speed
6. **Clear pixels**: Turn off pixels when moving animations to prevent trails
7. **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
This skill generates complete, runnable MicroPython programs for the Moving Rainbow LED strip project on a Raspberry Pi Pico. It produces clear, educational code that uses a config module for hardware settings, includes button controls, and demonstrates common NeoPixel animations and interaction patterns. Programs are structured for progressive learning and easy modification.
The generator outputs MicroPython files that import machine, neopixel, utime, and a config module. It initializes the NeoPixel strip using config values, provides reusable functions (wheel, erase, move_dot, color_wipe, rainbow_cycle, comet_tail, etc.), and builds a main loop with optional button-driven mode switching and debouncing. Comments, constants, and print statements are included so students can follow and tweak behavior.
Can I change the NeoPixel data pin or number of pixels?
Yes. Update the values in the config.py file and the generated program will use them automatically.
How are button presses handled to avoid bouncing?
Generated handlers implement a 200ms debounce using ticks_ms() and ignore events that occur sooner.