Use old approach, write directly to DrawTarget
This commit is contained in:
parent
95c7f00e20
commit
802504ff9d
3 changed files with 50 additions and 41 deletions
|
@ -7,9 +7,10 @@ use lvgl::display::{Display, DisplayBuffer, DisplayDriver};
|
||||||
type ColorSpace = Rgb565;
|
type ColorSpace = Rgb565;
|
||||||
|
|
||||||
fn main() {
|
fn main() {
|
||||||
let mut embedded_graphics_display: SimulatorDisplay<ColorSpace> = SimulatorDisplay::new(
|
let embedded_graphics_display: SimulatorDisplay<ColorSpace> = SimulatorDisplay::new(Size::new(
|
||||||
Size::new(lvgl_sys::LV_HOR_RES_MAX, lvgl_sys::LV_VER_RES_MAX),
|
lvgl_sys::LV_HOR_RES_MAX,
|
||||||
);
|
lvgl_sys::LV_VER_RES_MAX,
|
||||||
|
));
|
||||||
|
|
||||||
let output_settings = OutputSettingsBuilder::new().scale(2).build();
|
let output_settings = OutputSettingsBuilder::new().scale(2).build();
|
||||||
let mut window = Window::new("App Example", &output_settings);
|
let mut window = Window::new("App Example", &output_settings);
|
||||||
|
@ -17,10 +18,6 @@ fn main() {
|
||||||
// LVGL usage
|
// LVGL usage
|
||||||
lvgl::init();
|
lvgl::init();
|
||||||
|
|
||||||
let mut display_diver: DisplayDriver<ColorSpace> =
|
let mut display_diver = DisplayDriver::new(DisplayBuffer::new(), embedded_graphics_display);
|
||||||
DisplayDriver::new(DisplayBuffer::new(), |pixels| {
|
let gui = lvgl::disp_drv_register(&mut display_diver).unwrap();
|
||||||
// Here we draw to the external display
|
|
||||||
let _ = embedded_graphics_display.draw_iter(pixels);
|
|
||||||
});
|
|
||||||
let _display = lvgl::disp_drv_register(&mut display_diver).unwrap();
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,7 @@ const REFRESH_BUFFER_LEN: usize = 2;
|
||||||
// Declare a buffer for the refresh rate
|
// Declare a buffer for the refresh rate
|
||||||
pub(crate) const BUF_SIZE: usize = lvgl_sys::LV_HOR_RES_MAX as usize * REFRESH_BUFFER_LEN;
|
pub(crate) const BUF_SIZE: usize = lvgl_sys::LV_HOR_RES_MAX as usize * REFRESH_BUFFER_LEN;
|
||||||
|
|
||||||
|
#[derive(Debug, Copy, Clone)]
|
||||||
pub enum DisplayError {
|
pub enum DisplayError {
|
||||||
FailedToRegister,
|
FailedToRegister,
|
||||||
NotRegistered,
|
NotRegistered,
|
||||||
|
@ -61,67 +62,64 @@ impl DisplayBuffer {
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Copy, Clone)]
|
#[derive(Copy, Clone)]
|
||||||
pub struct DisplayDriver<C>
|
pub struct DisplayDriver<T, C>
|
||||||
where
|
where
|
||||||
|
T: DrawTarget<C>,
|
||||||
C: PixelColor + From<Color>,
|
C: PixelColor + From<Color>,
|
||||||
{
|
{
|
||||||
pub(crate) disp_drv: lvgl_sys::lv_disp_drv_t,
|
pub(crate) disp_drv: lvgl_sys::lv_disp_drv_t,
|
||||||
phantom: PhantomData<C>,
|
phantom_display: PhantomData<T>,
|
||||||
|
phantom_color: PhantomData<C>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<C> DisplayDriver<C>
|
impl<T, C> DisplayDriver<T, C>
|
||||||
where
|
where
|
||||||
|
T: DrawTarget<C>,
|
||||||
C: PixelColor + From<Color>,
|
C: PixelColor + From<Color>,
|
||||||
{
|
{
|
||||||
pub fn new<I, F>(display_buffer: DisplayBuffer, callback: F) -> Self
|
pub fn new(display_buffer: DisplayBuffer, native_display: T) -> Self {
|
||||||
where
|
|
||||||
I: IntoIterator<Item = drawable::Pixel<C>>,
|
|
||||||
F: FnMut(I) + 'static,
|
|
||||||
{
|
|
||||||
let mut disp_buf = ManuallyDrop::new(display_buffer.disp_buf);
|
let mut disp_buf = ManuallyDrop::new(display_buffer.disp_buf);
|
||||||
let mut callback = ManuallyDrop::new(callback);
|
let mut native_display = ManuallyDrop::new(DisplayUserData {
|
||||||
|
display: native_display,
|
||||||
|
phantom: PhantomData,
|
||||||
|
});
|
||||||
let mut disp_drv = unsafe {
|
let mut disp_drv = unsafe {
|
||||||
let mut disp_drv = MaybeUninit::uninit();
|
let mut disp_drv = MaybeUninit::uninit();
|
||||||
lvgl_sys::lv_disp_drv_init(disp_drv.as_mut_ptr());
|
lvgl_sys::lv_disp_drv_init(disp_drv.as_mut_ptr());
|
||||||
disp_drv.assume_init()
|
disp_drv.assume_init()
|
||||||
};
|
};
|
||||||
disp_drv.buffer = &mut *disp_buf as *mut lvgl_sys::lv_disp_buf_t;
|
disp_drv.buffer = &mut *disp_buf as *mut lvgl_sys::lv_disp_buf_t;
|
||||||
disp_drv.user_data = &mut callback as *mut _ as lvgl_sys::lv_disp_drv_user_data_t;
|
disp_drv.user_data = &mut native_display as *mut _ as lvgl_sys::lv_disp_drv_user_data_t;
|
||||||
disp_drv.flush_cb = Some(disp_flush_trampoline::<C, I, F>);
|
disp_drv.flush_cb = Some(disp_flush_trampoline::<T, C>);
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
disp_drv,
|
disp_drv,
|
||||||
phantom: PhantomData,
|
phantom_color: PhantomData,
|
||||||
|
phantom_display: PhantomData,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// impl Default for DisplayDriver {
|
pub(crate) struct DisplayUserData<T, C>
|
||||||
// fn default() -> Self {
|
where
|
||||||
// Self::new(DisplayBuffer::new())
|
T: DrawTarget<C>,
|
||||||
// }
|
C: PixelColor + From<Color>,
|
||||||
// }
|
{
|
||||||
|
display: T,
|
||||||
pub struct DisplayDriverBuilder {
|
phantom: PhantomData<C>,
|
||||||
disp_buf: Option<DisplayBuffer>,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl DisplayDriverBuilder {
|
unsafe extern "C" fn disp_flush_trampoline<T, C>(
|
||||||
pub fn with_callback() {}
|
|
||||||
}
|
|
||||||
|
|
||||||
unsafe extern "C" fn disp_flush_trampoline<C, I, F>(
|
|
||||||
disp_drv: *mut lvgl_sys::lv_disp_drv_t,
|
disp_drv: *mut lvgl_sys::lv_disp_drv_t,
|
||||||
area: *const lvgl_sys::lv_area_t,
|
area: *const lvgl_sys::lv_area_t,
|
||||||
color_p: *mut lvgl_sys::lv_color_t,
|
color_p: *mut lvgl_sys::lv_color_t,
|
||||||
) where
|
) where
|
||||||
|
T: DrawTarget<C>,
|
||||||
C: PixelColor + From<Color>,
|
C: PixelColor + From<Color>,
|
||||||
I: IntoIterator<Item = drawable::Pixel<C>>,
|
|
||||||
F: FnMut(I) + 'static,
|
|
||||||
{
|
{
|
||||||
let display_driver = *disp_drv;
|
let display_driver = *disp_drv;
|
||||||
if !display_driver.user_data.is_null() {
|
if !display_driver.user_data.is_null() {
|
||||||
let callback = &mut *(display_driver.user_data as *mut F);
|
let user_data = &mut *(display_driver.user_data as *mut DisplayUserData<T, C>);
|
||||||
let x1 = (*area).x1;
|
let x1 = (*area).x1;
|
||||||
let x2 = (*area).x2;
|
let x2 = (*area).x2;
|
||||||
let y1 = (*area).y1;
|
let y1 = (*area).y1;
|
||||||
|
@ -140,11 +138,25 @@ unsafe extern "C" fn disp_flush_trampoline<C, I, F>(
|
||||||
let color_len = x_len * iy + ix;
|
let color_len = x_len * iy + ix;
|
||||||
let lv_color = unsafe { *color_p.add(color_len) };
|
let lv_color = unsafe { *color_p.add(color_len) };
|
||||||
let raw_color = Color::from_raw(lv_color);
|
let raw_color = Color::from_raw(lv_color);
|
||||||
drawable::Pixel(Point::new(x as i32, y as i32), raw_color.into())
|
drawable::Pixel::<C>(Point::new(x as i32, y as i32), raw_color.into())
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
.flatten();
|
.flatten();
|
||||||
|
|
||||||
callback(pixels);
|
let _ = user_data.display.draw_iter(pixels);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// impl Default for DisplayDriver {
|
||||||
|
// fn default() -> Self {
|
||||||
|
// Self::new(DisplayBuffer::new())
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
|
||||||
|
pub struct DisplayDriverBuilder {
|
||||||
|
disp_buf: Option<DisplayBuffer>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl DisplayDriverBuilder {
|
||||||
|
pub fn with_callback() {}
|
||||||
|
}
|
||||||
|
|
|
@ -4,8 +4,8 @@ use core::ptr::NonNull;
|
||||||
use embedded_graphics::drawable;
|
use embedded_graphics::drawable;
|
||||||
use embedded_graphics::prelude::*;
|
use embedded_graphics::prelude::*;
|
||||||
|
|
||||||
pub fn disp_drv_register<C: PixelColor + From<Color>>(
|
pub fn disp_drv_register<C: PixelColor + From<Color>, T: DrawTarget<C>>(
|
||||||
disp_drv: &mut DisplayDriver<C>,
|
disp_drv: &mut DisplayDriver<T, C>,
|
||||||
) -> Result<Display, DisplayError> {
|
) -> Result<Display, DisplayError> {
|
||||||
let disp_ptr = unsafe {
|
let disp_ptr = unsafe {
|
||||||
lvgl_sys::lv_disp_drv_register(&mut disp_drv.disp_drv as *mut lvgl_sys::lv_disp_drv_t)
|
lvgl_sys::lv_disp_drv_register(&mut disp_drv.disp_drv as *mut lvgl_sys::lv_disp_drv_t)
|
||||||
|
|
Loading…
Reference in a new issue