slint/sixtyfps_runtime/corelib/item_rendering.rs
Simon Hausmann 33dd2d765d Minor cleanup in item transform calculation
We can use the new geometry() getter to simplify the transform update by origin.
2020-05-22 09:33:08 +02:00

100 lines
3.7 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, image);
rendering_data.cache_index = rendering_cache.allocate_entry(image_primitive);
rendering_data.cache_ok = true;
}
}
RenderingInfo::Text(_x, _y, text, font_family, font_pixel_size, color) => {
let pixel_size = if font_pixel_size != 0. { font_pixel_size } else { 48.0 * 72. / 96. };
let primitive = rendering_primitives_builder.create_glyphs(
&text,
&font_family,
pixel_size,
Color::from_argb_encoded(color),
);
rendering_data.cache_index = rendering_cache.allocate_entry(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 origin = item.geometry().origin;
let transform =
transform * Matrix4::from_translation(Vector3::new(origin.x, origin.y, 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,
);
}