from machine import Pin
import time
import math
import urandom
import random
import neopixel

ws_pin = 13
NUMPIXELS = 24         # NeoPixel ring size

pixDELAY  =  100         # Delay for pixel persistence
GRN_Base = (NUMPIXELS // 4)
BLU_Base = (NUMPIXELS // 2)
YEL_Base = ((NUMPIXELS * 3) // 4)

# Function to generate a random color
def random_color():
  return (urandom.randint(0, 255), urandom.randint(0, 255), urandom.randint(0, 255))

# Function to update the flowing light effect for NeoPixel LED ring
def update_flow():
  for j in range(20):
    for i in range(NUMPIXELS):
      pixels[i] = random_color()
    pixels.write()
    time.sleep_ms(pixDELAY)

# Reference: https://wokwi.com/projects/405827644049669121
def walk(steps):
  for _ in range(steps):
    step()
    paint()
    time.sleep_ms(pixDELAY)

def step():
  global x
  global direction
  if direction == 'left':
    x = max(0, x - 1)
    if x == 0:
      direction = 'right'
  elif direction == 'right':
    x = min(NUMPIXELS-1, x + 1)
    if x == NUMPIXELS-1:
      direction = 'left'
  for i in range(NUMPIXELS):
    ring[i] = max(0, ring[i] - 10)  # Fade out all positions
  ring[x] = 100  # Mark the new position

def paint():
  for i in range(NUMPIXELS):
    pixels[i] = (0, ring[i], 0)
  pixels.write() # Update the LED strip

# Reference: https://wokwi.com/projects/359664314470215681
def rainbow_animation():
  for rotation in range(NUMPIXELS * 10):  # Repeat the animation 10 times
    for index in range(NUMPIXELS):
      hue = (index + rotation) % NUMPIXELS / NUMPIXELS
      pixels[index] = hsv_to_rgb(hue, 1, 1)  # Set the current pixel to the rainbow color
    pixels.write()  # Update pixel colors
    time.sleep_ms(5)  # Wait between changes

def hsv_to_rgb(h, s, v):
  i = int(h * 6.)
  f = (h * 6.) - i
  p = v * (1. - s)
  q = v * (1. - s * f)
  t = v * (1. - s * (1. - f))
  i = i % 6

  if i == 0:
    return int(v * 255), int(t * 255), int(p * 255)
  elif i == 1:
    return int(q * 255), int(v * 255), int(p * 255)
  elif i == 2:
    return int(p * 255), int(v * 255), int(t * 255)
  elif i == 3:
    return int(p * 255), int(q * 255), int(v * 255)
  elif i == 4:
    return int(t * 255), int(p * 255), int(v * 255)
  elif i == 5:
    return int(v * 255), int(p * 255), int(q * 255)

# Reference: https://wokwi.com/projects/338857849104892500
# 彩虹滾動圈圈
def rainbowCycle(wait):
  for j in range(256 * 5):
    for i in range(NUMPIXELS):
      pixels[i] = Wheel(((i * 256 // NUMPIXELS) + j) & 255)
    pixels.write()
    time.sleep_ms(wait)

# 產生漸變顏色值
def  Wheel(WheelPos):
  WheelPos = 255 - WheelPos
  if (WheelPos < 85) :
    return [255 - WheelPos * 3, 0, WheelPos * 3]

  if (WheelPos < 170) :
    WheelPos = WheelPos - 85
    return [0, WheelPos * 3, 255 - WheelPos * 3]

  WheelPos = WheelPos - 170
  return [WheelPos * 3, 255 - WheelPos * 3, 0]

# PassingColors
# Reference: https://wokwi.com/projects/384675870918916097
# by xfpd: https://wokwi.com/makers/xfpd
def collide():
  global counter

  # RED pixel follows the counter (at the bottom of this function)
  RED = counter

  # GREEN pixel starts at 1/4 (.25) position on the ring
  GRN = GRN_Base - counter
  if (GRN < 0) :
    GRN = NUMPIXELS + GRN

  # BLUE pixels starts at 1/2 (.5) position on the ring
  BLU = BLU_Base + counter
  if (BLU > NUMPIXELS - 1):
    BLU = BLU - NUMPIXELS

  # YELLOW pixel starts at 3/4 (.75) position on the ring
  YEL = YEL_Base - counter
  if (YEL < 0):
    YEL = abs(NUMPIXELS + YEL)

  print("RED %5d | GRN %5d | BLU %5d | YEL %5d" %(RED, GRN, BLU, YEL))
  pixels[RED]=(255, 0, 0)
  pixels[GRN]=(0, 255, 0)
  pixels[BLU]=(0, 0, 255)
  pixels[YEL]=(255, 255, 0)
  pixels.write()
  time.sleep_ms(pixDELAY)

  # BLACK-out the trailing colored pixel
  pixels.fill((0, 0, 0))
  pixels.write()

  if (counter < (NUMPIXELS-1)):
    counter = counter + 1
  else:
    counter = 0

# Start Function
if __name__ == '__main__':
  print('NeoPixel LED Ring 彩虹特效')
  # Create a NeoPixel object
  pixels = neopixel.NeoPixel(Pin(ws_pin), NUMPIXELS)
  counter = 0 # counter

  while True:
    update_flow() # flowing light effect

    print('PassingColors')
    for j in range(NUMPIXELS * 5):
      collide()

    ring = [0 for _ in range(NUMPIXELS)]
    direction = random.choice(['left', 'right'])
    x = NUMPIXELS//4
    walk(NUMPIXELS*8)

    print('rainbow_animation()')
    rainbow_animation()  # Bonus "Easter egg" animation

    print('rainbowCycle()')
    rainbowCycle(1) # 彩虹滾動圈圈
BOOTSELLED1239USBRaspberryPiPico©2020RP2-8020/21P64M15.00TTT
pico:GP0
pico:GP1
pico:GND.1
pico:GP2
pico:GP3
pico:GP4
pico:GP5
pico:GND.2
pico:GP6
pico:GP7
pico:GP8
pico:GP9
pico:GND.3
pico:GP10
pico:GP11
pico:GP12
pico:GP13
pico:GND.4
pico:GP14
pico:GP15
pico:GP16
pico:GP17
pico:GND.5
pico:GP18
pico:GP19
pico:GP20
pico:GP21
pico:GND.6
pico:GP22
pico:RUN
pico:GP26
pico:GP27
pico:GND.7
pico:GP28
pico:ADC_VREF
pico:3V3
pico:3V3_EN
pico:GND.8
pico:VSYS
pico:VBUS
ring1:GND
ring1:VCC
ring1:DIN
ring1:DOUT