home / skills / dmccreary / claude-skills / moving-rainbow

moving-rainbow skill

/skills/moving-rainbow

This skill generates educational MicroPython programs for Moving Rainbow on Raspberry Pi Pico with NeoPixel strips and button controls.

npx playbooks add skill dmccreary/claude-skills --skill moving-rainbow

Review the files below or copy the command above to add this skill to your agents.

Files (5)
SKILL.md
7.6 KB
---
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

Overview

This skill generates complete MicroPython programs for the Moving Rainbow LED strip educational project on a Raspberry Pi Pico. It produces runnable code that uses a config.py hardware mapping, NeoPixel control, and optional button-driven mode switching with clear comments for learning.

How this skill works

Programs follow a consistent template: import machine/neopixel/utime and config, initialize the NeoPixel strip, define helper functions (wheel, erase, animations), and run an infinite loop to drive animations. Button handling uses interrupts with a 200ms debounce to change modes safely. The generator emphasizes readable names, adjustable constants, and visible feedback via print statements.

When to use it

  • You need a ready-to-run MicroPython example for a NeoPixel strip on Raspberry Pi Pico
  • Teaching students progressive LED animation concepts and patterns
  • Prototyping multiple animation modes switched by physical buttons
  • Creating educational examples that demonstrate color wheels, wipe, and comet effects
  • Generating code templates that use config.py for hardware mapping

Best practices

  • Always import and use config values (NEOPIXEL_PIN, NUMBER_PIXELS, BUTTON_PIN_x)
  • Call strip.write() after any pixel changes to update the LEDs
  • Use modulo arithmetic for wrapping indices: counter % config.NUMBER_PIXELS
  • Debounce button input by requiring at least 200ms between presses
  • Provide adjustable delay and tail-length constants so students can experiment
  • Include clear comments and print() statements to show program state

Example use cases

  • Generate a simple moving dot program that cycles a single pixel around the strip
  • Produce a rainbow cycle animation that smoothly shifts colors across all pixels
  • Create a multi-mode program where two buttons increment/decrement animation modes
  • Build a comet-tail effect with configurable tail length and brightness levels
  • Provide a random-color educational demo using urandom for pixel selection

FAQ

What hardware pins does generated code expect?

Generated programs use config.py values by default: NEOPIXEL_PIN=0, BUTTON_PIN_1=14, BUTTON_PIN_2=15, and the Pico built-in LED on 25.

How does button debouncing work in the examples?

Button handlers record ticks_ms() and ignore presses within 200ms of the last accepted press to avoid bounce and rapid mode changes.