use core::marker::PhantomData;
use embedded_graphics::{
pixelcolor::Rgb888,
prelude::{DrawTarget, PixelColor, Point, Size},
primitives::Rectangle,
};
use embedded_layout::{object_chain::ChainElement, prelude::*, view_group::ViewGroup};
use crate::{
interaction::InputAdapterSource,
selection_indicator::{style::IndicatorStyle, SelectionIndicatorController},
Marker, MenuItem, MenuStyle,
};
pub trait MenuItemCollection<R> {
fn bounds_of(&self, nth: usize) -> Rectangle;
fn title_of(&self, nth: usize) -> &str;
fn details_of(&self, nth: usize) -> &str;
fn interact_with(&mut self, nth: usize) -> R;
fn count(&self) -> usize;
fn draw_styled<C, S, IT, P, DIS>(
&self,
style: &MenuStyle<C, S, IT, P, R>,
display: &mut DIS,
) -> Result<(), DIS::Error>
where
C: PixelColor + From<Rgb888>,
S: IndicatorStyle,
IT: InputAdapterSource<R>,
P: SelectionIndicatorController,
DIS: DrawTarget<Color = C>;
}
impl<I, R> MenuItemCollection<R> for I
where
I: MenuItem<R> + Marker,
{
fn bounds_of(&self, nth: usize) -> Rectangle {
debug_assert!(nth == 0);
self.bounds()
}
fn interact_with(&mut self, nth: usize) -> R {
debug_assert!(nth == 0);
self.interact()
}
fn title_of(&self, nth: usize) -> &str {
debug_assert!(nth == 0);
self.title()
}
fn details_of(&self, nth: usize) -> &str {
debug_assert!(nth == 0);
self.details()
}
fn count(&self) -> usize {
1
}
fn draw_styled<C, S, IT, P, DIS>(
&self,
style: &MenuStyle<C, S, IT, P, R>,
display: &mut DIS,
) -> Result<(), DIS::Error>
where
C: PixelColor + From<Rgb888>,
S: IndicatorStyle,
IT: InputAdapterSource<R>,
P: SelectionIndicatorController,
DIS: DrawTarget<Color = C>,
{
MenuItem::draw_styled(self, style, display)
}
}
pub struct MenuItems<C, I, R>
where
C: AsRef<[I]> + AsMut<[I]>,
I: MenuItem<R>,
{
items: C,
position: Point,
_marker: PhantomData<(I, R)>,
}
impl<C, I, R> MenuItems<C, I, R>
where
C: AsRef<[I]> + AsMut<[I]>,
I: MenuItem<R>,
{
pub fn new(mut items: C) -> Self {
let mut offset = 0;
for item in items.as_mut().iter_mut() {
item.translate_mut(Point::new(0, offset));
offset += item.bounds().size.height as i32;
}
Self {
items,
position: Point::zero(),
_marker: PhantomData,
}
}
}
impl<C, I, R> MenuItemCollection<R> for MenuItems<C, I, R>
where
C: AsRef<[I]> + AsMut<[I]>,
I: MenuItem<R>,
{
fn bounds_of(&self, nth: usize) -> Rectangle {
self.items.as_ref()[nth].bounds()
}
fn interact_with(&mut self, nth: usize) -> R {
self.items.as_mut()[nth].interact()
}
fn title_of(&self, nth: usize) -> &str {
self.items.as_ref()[nth].title()
}
fn details_of(&self, nth: usize) -> &str {
self.items.as_ref()[nth].details()
}
fn count(&self) -> usize {
self.items.as_ref().len()
}
fn draw_styled<PC, S, IT, P, DIS>(
&self,
style: &MenuStyle<PC, S, IT, P, R>,
display: &mut DIS,
) -> Result<(), DIS::Error>
where
PC: PixelColor + From<Rgb888>,
S: IndicatorStyle,
IT: InputAdapterSource<R>,
P: SelectionIndicatorController,
DIS: DrawTarget<Color = PC>,
{
for item in self.items.as_ref() {
item.draw_styled(style, display)?;
}
Ok(())
}
}
impl<C, I, R> View for MenuItems<C, I, R>
where
C: AsRef<[I]> + AsMut<[I]>,
I: MenuItem<R>,
{
fn translate_impl(&mut self, by: Point) {
self.position += by;
for view in self.items.as_mut().iter_mut() {
view.translate_impl(by);
}
}
fn bounds(&self) -> Rectangle {
let items = &self.items.as_ref();
if items.is_empty() {
return Rectangle::new(self.position, Size::zero());
}
let mut rect = items[0].bounds();
for view in items[1..].iter() {
rect = rect.enveloping(&view.bounds());
}
rect
}
}
impl<C, I, R> ViewGroup for MenuItems<C, I, R>
where
C: AsRef<[I]> + AsMut<[I]>,
I: MenuItem<R>,
{
fn len(&self) -> usize {
self.count()
}
fn at(&self, idx: usize) -> &dyn View {
&self.items.as_ref()[idx]
}
fn at_mut(&mut self, idx: usize) -> &mut dyn View {
&mut self.items.as_mut()[idx]
}
}
impl<I, R> MenuItemCollection<R> for Chain<I>
where
I: MenuItemCollection<R>,
{
fn bounds_of(&self, nth: usize) -> Rectangle {
self.object.bounds_of(nth)
}
fn interact_with(&mut self, nth: usize) -> R {
self.object.interact_with(nth)
}
fn title_of(&self, nth: usize) -> &str {
self.object.title_of(nth)
}
fn details_of(&self, nth: usize) -> &str {
self.object.details_of(nth)
}
fn count(&self) -> usize {
self.object.count()
}
fn draw_styled<PC, S, IT, P, DIS>(
&self,
style: &MenuStyle<PC, S, IT, P, R>,
display: &mut DIS,
) -> Result<(), DIS::Error>
where
PC: PixelColor + From<Rgb888>,
S: IndicatorStyle,
IT: InputAdapterSource<R>,
P: SelectionIndicatorController,
DIS: DrawTarget<Color = PC>,
{
self.object.draw_styled(style, display)
}
}
impl<I, LE, R> MenuItemCollection<R> for Link<I, LE>
where
I: MenuItemCollection<R>,
LE: MenuItemCollection<R> + ChainElement,
{
fn bounds_of(&self, nth: usize) -> Rectangle {
let count = self.parent.count();
if nth < count {
self.parent.bounds_of(nth)
} else {
self.object.bounds_of(nth - count)
}
}
fn interact_with(&mut self, nth: usize) -> R {
let count = self.parent.count();
if nth < count {
self.parent.interact_with(nth)
} else {
self.object.interact_with(nth - count)
}
}
fn title_of(&self, nth: usize) -> &str {
let count = self.parent.count();
if nth < count {
self.parent.title_of(nth)
} else {
self.object.title_of(nth - count)
}
}
fn details_of(&self, nth: usize) -> &str {
let count = self.parent.count();
if nth < count {
self.parent.details_of(nth)
} else {
self.object.details_of(nth - count)
}
}
fn count(&self) -> usize {
self.object.count() + self.parent.count()
}
fn draw_styled<PC, S, IT, P, DIS>(
&self,
style: &MenuStyle<PC, S, IT, P, R>,
display: &mut DIS,
) -> Result<(), DIS::Error>
where
PC: PixelColor + From<Rgb888>,
S: IndicatorStyle,
IT: InputAdapterSource<R>,
P: SelectionIndicatorController,
DIS: DrawTarget<Color = PC>,
{
self.parent.draw_styled(style, display)?;
self.object.draw_styled(style, display)?;
Ok(())
}
}