#![feature(trait_upcasting)] #![allow(arithmetic_overflow)] use esp_idf_svc::hal::prelude::Peripherals; use ws2812_esp32_rmt_driver::lib_embedded_graphics::{LedPixelShape, Ws2812DrawTarget}; use embedded_graphics::{ prelude::*, }; use palette::Hsv; use palette::convert::IntoColorUnclamped; mod power; mod lib8; mod render; mod task; mod time; mod geometry; mod embedded_graphics_lib; use crate::time::Periodically; use crate::geometry::{Coordinates, VirtualCoordinates}; use crate::embedded_graphics_lib::EmbeddedDisplay; use crate::render::{Surfaces, Surface, SimpleSurface, Display}; use crate::task::Task; struct IdleTask { frame: u8, surface: T, updater: Periodically } struct IdleShader { frame: u8 } impl render::Shader for IdleShader { fn draw(&self, coords: VirtualCoordinates) -> lib8::RGB8 { Hsv::new_srgb(self.frame.wrapping_add(coords.x()).wrapping_add(coords.y()), 255, 255).into_color_unclamped() } } impl IdleTask { fn new(surface: T) -> Self { IdleTask { frame: 0, surface: surface, updater: Periodically::new_every_n_ms(16) } } } impl task::Task for IdleTask { fn name(&self) -> &'static str { "Idle" } fn tick(&mut self) { self.updater.run(|| { self.frame = self.frame.wrapping_add(1); self.surface.set_shader(Box::new(IdleShader { frame: self.frame })); }) } fn stop(&mut self) { self.surface.clear_shader(); } } struct PonderjarMatrix {} impl LedPixelShape for PonderjarMatrix { fn size() -> Size { Size::new(17, 17) } fn pixel_index(point: Point) -> Option { if (0..Self::size().width as i32).contains(&point.x) && (0..Self::size().height as i32).contains(&point.y) { if point.y % 2 == 0 { Some((point.y as u32 * Self::size().width as u32 + point.x as u32).try_into().unwrap()) } else { Some((point.y as u32 * Self::size().width as u32 - point.x as u32).try_into().unwrap()) } } else { None } } } type PonderjarTarget<'a> = Ws2812DrawTarget<'a, PonderjarMatrix>; trait DisplayInit { fn new_display() -> impl Display + Task; } impl DisplayInit for Ws2812DrawTarget<'_, Shape> { fn new_display() -> impl Display + Task { let peripherals = Peripherals::take().unwrap(); let led_pin = peripherals.pins.gpio14; let channel = peripherals.rmt.channel0; const POWER_VOLTS : u32 = 5; const POWER_MA : u32 = 500; const MAX_POWER_MW : u32 = POWER_VOLTS * POWER_MA; let target = Self::new(channel, led_pin).unwrap(); return EmbeddedDisplay::::new(target, MAX_POWER_MW); } } fn main() { // It is necessary to call this function once. Otherwise some patches to the runtime // implemented by esp-idf-sys might not link properly. See https://github.com/esp-rs/esp-idf-template/issues/71 esp_idf_svc::sys::link_patches(); // Bind the log crate to the ESP Logging facilities esp_idf_svc::log::EspLogger::initialize_default(); log::info!("Setting up display"); let mut display = PonderjarTarget::new_display::(); log::info!("Creating runner"); let mut runner = task::Scheduler::new(vec![ Box::new(IdleTask::new(display.new_surface().unwrap())), Box::new(display), ]); log::info!("Ready to rock and roll"); loop { runner.tick(); } }