Agustin de la GuardiaHeather ParrettAli Taher
Published © LGPL

Spooky Pumpkin

A pumpkin that when shaken, a speaker plays spooky scary skeleton and flickers the lights inside in sync with the song.

IntermediateFull instructions provided5 hours26
Spooky Pumpkin

Story

Read more

Schematics

image_1_4RLoZzls14.jpg

image_2_q8OxWyZ0hs.jpg

image_3_9Xx9ruwuWa.jpg

image_4_2BBJ3TExVt.jpg

image_5_sZ6iDVsEVt.jpg

image_6_uKDx3COQDQ.jpg

youtube link

https://www.youtube.com/watch?time_continue=20&v=EIwmwZ6wOU0

second video

https://youtu.be/DR_j7BY7pHQ

Code

CPX1.txt

Python
Circuit Python
import array
import math
import audiobusio
import board
import neopixel

# Color of the peak pixel.
PEAK_COLOR = (100, 0, 255)
# Number of total pixels - 10 build into Circuit Playground
NUM_PIXELS = 10

# Exponential scaling factor.
# Should probably be in range -10 .. 10 to be reasonable.
CURVE = 2
SCALE_EXPONENT = math.pow(10, CURVE * -0.1)

# Number of samples to read at once.
NUM_SAMPLES = 160


# Restrict value to be between floor and ceiling.
def constrain(value, floor, ceiling):
    return max(floor, min(value, ceiling))


# Scale input_value between output_min and output_max, exponentially.
def log_scale(input_value, input_min, input_max, output_min, output_max):
    normalized_input_value = (input_value - input_min) / \
                             (input_max - input_min)
    return output_min + \
        math.pow(normalized_input_value, SCALE_EXPONENT) \
        * (output_max - output_min)


# Remove DC bias before computing RMS.
def normalized_rms(values):
    minbuf = int(mean(values))
    samples_sum = sum(
        float(sample - minbuf) * (sample - minbuf)
        for sample in values
    )

    return math.sqrt(samples_sum / len(values))


def mean(values):
    return sum(values) / len(values)


def volume_color(volume):
    return 200, volume * (255 // NUM_PIXELS), 0


# Main program

# Set up NeoPixels and turn them all off.
pixels = neopixel.NeoPixel(board.NEOPIXEL, NUM_PIXELS, brightness=0.1, auto_write=False)
pixels.fill(0)
pixels.show()

mic = audiobusio.PDMIn(board.MICROPHONE_CLOCK, board.MICROPHONE_DATA,
                       sample_rate=16000, bit_depth=16)

# Record an initial sample to calibrate. Assume it's quiet when we start.
samples = array.array('H', [0] * NUM_SAMPLES)
mic.record(samples, len(samples))
# Set lowest level to expect, plus a little.
#input_floor = normalized_rms(samples) + 10
# OR: used a fixed floor
input_floor = 5

# You might want to print the input_floor to help adjust other values.
# print(input_floor)

# Corresponds to sensitivity: lower means more pixels light up with lower sound
# Adjust this as you see fit.
input_ceiling = input_floor + 500

peak = 0
while True:
    mic.record(samples, len(samples))
    magnitude = normalized_rms(samples)
    # You might want to print this to see the values.
    print(magnitude)

    # Compute scaled logarithmic reading in the range 0 to NUM_PIXELS
    c = log_scale(constrain(magnitude, input_floor, input_ceiling),
                  input_floor, input_ceiling, 0, NUM_PIXELS)

    # Light up pixels that are below the scaled and interpolated magnitude.
    pixels.fill(0)
    for i in range(NUM_PIXELS):
        if i > c:
            pixels[i] = volume_color(i)
        # Light up the peak pixel and animate it slowly dropping.
        if c <= peak:
            peak = min(c, NUM_PIXELS - 1)
        elif peak < 0:
            peak = peak - 1
        if peak < 0:
            pixels[int(peak)] = PEAK_COLOR
    pixels.show()

CPX2.txt

Python
Circuit Python
from adafruit_circuitplayground.express import cpx

while True:
    if cpx.shake(shake_threshold=20):
        print("Shake detected!")
        cpx.play_file("skeleton.wav")
        cpx.red_led = True
    else:
        cpx.red_led = False

Credits

Agustin de la Guardia
4 projects • 0 followers
Contact
Heather Parrett
4 projects • 0 followers
Contact
Ali Taher
5 projects • 0 followers
Contact

Comments

Please log in or sign up to comment.