pimoroni-pico/micropython/examples/breakout_encoder_wheel/clock.py

98 lines
3.3 KiB
Python

import time
from machine import RTC
from pimoroni_i2c import PimoroniI2C
from pimoroni import BREAKOUT_GARDEN_I2C_PINS # or PICO_EXPLORER_I2C_PINS or HEADER_I2C_PINS
from breakout_encoder_wheel import BreakoutEncoderWheel, NUM_LEDS
"""
Displays a 12 hour clock on Encoder Wheel's LED ring, getting time from the system.
Press Ctrl+C to stop the program.
"""
# Datetime Indices
HOUR = 4
MINUTE = 5
SECOND = 6
# Constants
BRIGHTNESS = 1.0 # The brightness of the LEDs
UPDATES = 50 # How many times the LEDs will be updated per second
UPDATE_RATE_US = 1000000 // UPDATES
# Handy values for the number of milliseconds
MILLIS_PER_SECOND = 1000
MILLIS_PER_MINUTE = MILLIS_PER_SECOND * 60
MILLIS_PER_HOUR = MILLIS_PER_MINUTE * 60
MILLIS_PER_HALF_DAY = MILLIS_PER_HOUR * 12
# Create a new BreakoutEncoderWheel
i2c = PimoroniI2C(**BREAKOUT_GARDEN_I2C_PINS)
wheel = BreakoutEncoderWheel(i2c)
# Access the built-in RTC
rtc = RTC()
# Sleep until a specific time in the future. Use this instead of time.sleep() to correct for
# inconsistent timings when dealing with complex operations or external communication
def sleep_until(end_time):
time_to_sleep = time.ticks_diff(end_time, time.ticks_us())
if time_to_sleep > 0:
time.sleep_us(time_to_sleep)
# Simple function to clamp a value between a minimum and maximum
def clamp(n, smallest, largest):
return max(smallest, min(n, largest))
# Calculates the brightness of an LED based on its index and a position along the LED ring
def led_brightness_at(led, position, half_width=1, span=1):
brightness = 0.0
upper = position + half_width
lower = position - half_width
if led > position:
brightness = clamp((upper - led) / span, 0.0, 1.0)
else:
brightness = clamp((led - lower) / span, 0.0, 1.0)
# Handle the LEDs being in a circle
if upper >= NUM_LEDS:
brightness = clamp(((upper - NUM_LEDS) - led) / span, brightness, 1.0)
elif lower < 0.0:
brightness = clamp((led - (lower + NUM_LEDS)) / span, brightness, 1.0)
return int(brightness * BRIGHTNESS * 255)
# Make rainbows
while True:
# Record the start time of this loop
start_time = time.ticks_us()
# Get the current system time
now = rtc.datetime()
# Convert the seconds, minutes, and hours into milliseconds (this is done to give a smoother animation, particularly for the seconds hand)
sec_as_millis = (now[SECOND] * MILLIS_PER_SECOND)
min_as_millis = (now[MINUTE] * MILLIS_PER_MINUTE) + sec_as_millis
hour_as_millis = ((now[HOUR] % 12) * MILLIS_PER_HOUR) + min_as_millis
# Calculate the position on the LED ring that the, second, minute, and hour hands should be
sec_pos = min(sec_as_millis / MILLIS_PER_MINUTE, 1.0) * NUM_LEDS
min_pos = min(min_as_millis / MILLIS_PER_HOUR, 1.0) * NUM_LEDS
hour_pos = min(hour_as_millis / MILLIS_PER_HALF_DAY, 1.0) * NUM_LEDS
for i in range(NUM_LEDS):
# Turn on the LEDs close to the position of the current second, minute, and hour
r = led_brightness_at(i, sec_pos)
g = led_brightness_at(i, min_pos)
b = led_brightness_at(i, hour_pos)
wheel.set_rgb(i, r, g, b)
wheel.show()
# Sleep until the next update, accounting for how long the above operations took to perform
sleep_until(time.ticks_add(start_time, UPDATE_RATE_US))