#![warn(missing_docs)]
use super::graphics::RenderingCache;
use super::items::*;
use crate::component::ComponentRc;
use crate::graphics::Rect;
use crate::item_tree::ItemVisitorResult;
use core::pin::Pin;
use std::cell::{Cell, RefCell};
#[derive(Default, Debug)]
#[repr(C)]
pub struct CachedRenderingData {
pub(crate) cache_index: Cell<usize>,
pub(crate) cache_ok: Cell<bool>,
}
impl CachedRenderingData {
pub fn ensure_up_to_date<T: Clone>(
&self,
cache: &mut RenderingCache<T>,
update_fn: impl FnOnce() -> T,
) -> T {
if self.cache_ok.get() {
let index = self.cache_index.get();
let existing_entry = cache.get_mut(index).unwrap();
if let Some(data) =
existing_entry.dependency_tracker.as_ref().evaluate_if_dirty(update_fn)
{
existing_entry.data = data
}
existing_entry.data.clone()
} else {
self.cache_index.set(cache.insert(crate::graphics::CachedGraphicsData::new(update_fn)));
self.cache_ok.set(true);
cache.get(self.cache_index.get()).unwrap().data.clone()
}
}
pub fn release<T>(&self, cache: &mut RenderingCache<T>) {
if self.cache_ok.get() {
let index = self.cache_index.get();
cache.remove(index);
self.cache_ok.set(false);
}
}
}
pub fn render_component_items(
component: &ComponentRc,
renderer: &mut dyn ItemRenderer,
origin: crate::graphics::Point,
) {
let renderer = RefCell::new(renderer);
{
let mut renderer = renderer.borrow_mut();
renderer.save_state();
renderer.translate(origin.x, origin.y);
}
crate::item_tree::visit_items_with_post_visit(
component,
crate::item_tree::TraversalOrder::BackToFront,
|_, item, _, _| {
renderer.borrow_mut().save_state();
let item_geometry = item.as_ref().geometry();
let item_origin = item_geometry.origin;
if !renderer.borrow().get_current_clip().intersects(&item_geometry)
&& ItemRef::downcast_pin::<Flickable>(item).is_none()
&& ItemRef::downcast_pin::<Clip>(item).is_none()
{
renderer.borrow_mut().translate(item_origin.x, item_origin.y);
return (ItemVisitorResult::Continue(()), ());
}
renderer.borrow_mut().translate(item_origin.x, item_origin.y);
item.as_ref().render(&mut (*renderer.borrow_mut() as &mut dyn ItemRenderer));
(ItemVisitorResult::Continue(()), ())
},
|_, _, _, r| {
renderer.borrow_mut().restore_state();
r
},
(),
);
renderer.borrow_mut().restore_state();
}
#[allow(missing_docs)]
pub trait ItemRenderer {
fn draw_rectangle(&mut self, rect: Pin<&Rectangle>);
fn draw_border_rectangle(&mut self, rect: Pin<&BorderRectangle>);
fn draw_image(&mut self, image: Pin<&Image>);
fn draw_clipped_image(&mut self, image: Pin<&ClippedImage>);
fn draw_text(&mut self, text: Pin<&Text>);
fn draw_text_input(&mut self, text_input: Pin<&TextInput>);
fn draw_path(&mut self, path: Pin<&Path>);
fn draw_box_shadow(&mut self, box_shadow: Pin<&BoxShadow>);
fn combine_clip(&mut self, rect: Rect, radius: f32, border_width: f32);
fn get_current_clip(&self) -> Rect;
fn translate(&mut self, x: f32, y: f32);
fn rotate(&mut self, angle_in_degrees: f32);
fn apply_opacity(&mut self, opacity: f32);
fn save_state(&mut self);
fn restore_state(&mut self);
fn scale_factor(&self) -> f32;
fn draw_cached_pixmap(
&mut self,
item_cache: &CachedRenderingData,
update_fn: &dyn Fn(&mut dyn FnMut(u32, u32, &[u8])),
);
fn as_any(&mut self) -> &mut dyn core::any::Any;
}