slint/sixtyfps_runtime/corelib/item_rendering.rs
Simon Hausmann e6be2c91b8 Fix translation of children of images
Propagate the x/y of images also to the children.
2020-05-18 16:01:53 +02:00

95 lines
3.5 KiB
Rust

use super::abi::datastructures::{ItemRefMut, RenderingInfo};
use super::graphics::{Color, Frame, GraphicsBackend, RenderingCache, RenderingPrimitivesBuilder};
use cgmath::{Matrix4, SquareMatrix, Vector3};
use lyon::math::{Point, Rect, Size};
pub(crate) fn update_item_rendering_data<Backend: GraphicsBackend>(
mut item: ItemRefMut<'_>,
rendering_cache: &mut RenderingCache<Backend>,
rendering_primitives_builder: &mut Backend::RenderingPrimitivesBuilder,
) {
let item_rendering_info = item.rendering_info();
println!("Caching ... {:?}", item_rendering_info);
let rendering_data = item.cached_rendering_data_offset_mut();
match item_rendering_info {
RenderingInfo::Rectangle(_x, _y, width, height, color) => {
if width <= 0. || height <= 0. {
return;
}
let primitive = rendering_primitives_builder.create_rect_primitive(
width,
height,
Color::from_argb_encoded(color),
);
rendering_data.cache_index = rendering_cache.allocate_entry(primitive);
rendering_data.cache_ok = true;
}
RenderingInfo::Image(_x, _y, _source) => {
rendering_data.cache_ok = false;
#[cfg(not(target_arch = "wasm32"))]
{
let mut image_path = std::env::current_exe().unwrap();
image_path.pop(); // pop of executable name
image_path.push(_source);
let image = image::open(image_path.as_path()).unwrap().into_rgba();
let source_size = image.dimensions();
let rect = Rect::new(
Point::new(0.0, 0.0),
Size::new(source_size.0 as f32, source_size.1 as f32),
);
let image_primitive =
rendering_primitives_builder.create_image_primitive(rect, rect, image);
rendering_data.cache_index = rendering_cache.allocate_entry(image_primitive);
rendering_data.cache_ok = true;
}
}
RenderingInfo::NoContents => {
rendering_data.cache_ok = false;
}
}
}
pub(crate) fn render_component_items<Backend: GraphicsBackend>(
component: vtable::VRef<'_, crate::abi::datastructures::ComponentVTable>,
frame: &mut Backend::Frame,
rendering_cache: &RenderingCache<Backend>,
) {
let transform = Matrix4::identity();
crate::abi::datastructures::visit_items(
component,
|item, transform| {
let mut transform = transform.clone();
let item_rendering_info = item.rendering_info();
if let Some((x, y)) = match item_rendering_info {
RenderingInfo::Rectangle(x, y, ..) => Some((x, y)),
RenderingInfo::Image(x, y, ..) => Some((x, y)),
_ => None,
} {
transform = transform * Matrix4::from_translation(Vector3::new(x, y, 0.0));
}
let cached_rendering_data = item.cached_rendering_data_offset();
if cached_rendering_data.cache_ok {
println!(
"Rendering... {:?} from cache {}",
item_rendering_info, cached_rendering_data.cache_index
);
let primitive = rendering_cache.entry_at(cached_rendering_data.cache_index);
frame.render_primitive(&primitive, &transform);
}
transform
},
transform,
);
}