use std::cell::RefCell;
use std::pin::Pin;
use std::rc::Rc;
use euclid::approxeq::ApproxEq;
use i_slint_core::graphics::boxshadowcache::BoxShadowCache;
use i_slint_core::graphics::euclid::num::Zero;
use i_slint_core::graphics::euclid::{self};
use i_slint_core::graphics::rendering_metrics_collector::RenderingMetrics;
use i_slint_core::graphics::{IntRect, Point, Size};
use i_slint_core::item_rendering::{
CachedRenderingData, ItemCache, ItemRenderer, RenderBorderRectangle, RenderImage, RenderText,
};
use i_slint_core::items::{
self, Clip, FillRule, ImageRendering, ItemRc, Layer, Opacity, RenderingResult, TextStrokeStyle,
};
use i_slint_core::lengths::{
LogicalBorderRadius, LogicalLength, LogicalPoint, LogicalRect, LogicalSize, LogicalVector,
RectLengths, ScaleFactor, SizeLengths,
};
use i_slint_core::window::WindowInner;
use i_slint_core::{Brush, Color, ImageInner, SharedString};
use super::images::{Texture, TextureCacheKey};
use super::PhysicalSize;
use super::{fonts, PhysicalBorderRadius, PhysicalLength, PhysicalPoint, PhysicalRect};
type FemtovgBoxShadowCache = BoxShadowCache<ItemGraphicsCacheEntry>;
pub type Canvas = femtovg::Canvas<femtovg::renderer::OpenGl>;
pub type CanvasRc = Rc<RefCell<Canvas>>;
#[derive(Clone)]
pub enum ItemGraphicsCacheEntry {
Texture(Rc<Texture>),
ColorizedImage {
_original_image: Rc<Texture>,
colorized_image: Rc<Texture>,
},
}
impl ItemGraphicsCacheEntry {
fn as_texture(&self) -> &Rc<Texture> {
match self {
ItemGraphicsCacheEntry::Texture(image) => image,
ItemGraphicsCacheEntry::ColorizedImage { colorized_image, .. } => colorized_image,
}
}
fn is_colorized_image(&self) -> bool {
matches!(self, ItemGraphicsCacheEntry::ColorizedImage { .. })
}
}
pub(super) type ItemGraphicsCache = ItemCache<Option<ItemGraphicsCacheEntry>>;
const KAPPA90: f32 = 0.55228;
#[derive(Clone)]
struct State {
scissor: LogicalRect,
global_alpha: f32,
current_render_target: femtovg::RenderTarget,
}
pub struct GLItemRenderer<'a> {
graphics_cache: &'a ItemGraphicsCache,
texture_cache: &'a RefCell<super::images::TextureCache>,
box_shadow_cache: FemtovgBoxShadowCache,
canvas: CanvasRc,
textures_to_delete_after_flush: RefCell<Vec<Rc<super::images::Texture>>>,
window: &'a i_slint_core::api::Window,
scale_factor: ScaleFactor,
state: Vec<State>,
metrics: RenderingMetrics,
}
fn rect_with_radius_to_path(
rect: PhysicalRect,
border_radius: PhysicalBorderRadius,
) -> femtovg::Path {
let mut path = femtovg::Path::new();
let x = rect.origin.x;
let y = rect.origin.y;
let width = rect.size.width;
let height = rect.size.height;
if let Some(border_radius) = border_radius.as_uniform() {
if width.approx_eq(&height) && (border_radius * 2.).approx_eq(&width) {
path.circle(x + border_radius, y + border_radius, border_radius);
} else {
path.rounded_rect(x, y, width, height, border_radius);
}
} else {
path.rounded_rect_varying(
x,
y,
width,
height,
border_radius.top_left,
border_radius.top_right,
border_radius.bottom_right,
border_radius.bottom_left,
);
}
path
}
fn rect_to_path(r: PhysicalRect) -> femtovg::Path {
rect_with_radius_to_path(r, PhysicalBorderRadius::default())
}
fn adjust_rect_and_border_for_inner_drawing(
rect: &mut PhysicalRect,
border_width: &mut PhysicalLength,
) {
*border_width = border_width.min(rect.width_length() / 2.);
rect.origin += PhysicalSize::from_lengths(*border_width / 2., *border_width / 2.);
rect.size -= PhysicalSize::from_lengths(*border_width, *border_width);
}
fn path_bounding_box(canvas: &CanvasRc, path: &femtovg::Path) -> euclid::default::Box2D<f32> {
let mut canvas = canvas.borrow_mut();
canvas.save();
canvas.reset_transform();
let bounding_box = canvas.path_bbox(path);
canvas.restore();
euclid::default::Box2D::new(
[bounding_box.minx, bounding_box.miny].into(),
[bounding_box.maxx, bounding_box.maxy].into(),
)
}
fn clip_path_for_rect_alike_item(
clip_rect: LogicalRect,
mut radius: LogicalBorderRadius,
mut border_width: LogicalLength,
scale_factor: ScaleFactor,
) -> femtovg::Path {
radius -= LogicalBorderRadius::new_uniform(border_width.get() * KAPPA90);
border_width *= 2.;
let mut border_width = border_width * scale_factor;
let radius = radius * scale_factor;
let mut clip_rect = clip_rect * scale_factor;
adjust_rect_and_border_for_inner_drawing(&mut clip_rect, &mut border_width);
rect_with_radius_to_path(clip_rect, radius)
}
impl<'a> GLItemRenderer<'a> {
pub fn global_alpha_transparent(&self) -> bool {
self.state.last().unwrap().global_alpha == 0.0
}
pub fn draw_rect(&mut self, size: LogicalSize, brush: Brush) {
let geometry = PhysicalRect::from(size * self.scale_factor);
if geometry.is_empty() {
return;
}
if self.global_alpha_transparent() {
return;
}
let path = rect_to_path(geometry);
let paint = match self.brush_to_paint(brush, &path) {
Some(paint) => paint,
None => return,
}
.with_anti_alias(false);
self.canvas.borrow_mut().fill_path(&path, &paint);
}
}
impl<'a> ItemRenderer for GLItemRenderer<'a> {
fn draw_rectangle(&mut self, rect: Pin<&items::Rectangle>, _: &ItemRc, size: LogicalSize) {
self.draw_rect(size, rect.background());
}
fn draw_border_rectangle(
&mut self,
rect: Pin<&dyn RenderBorderRectangle>,
_: &ItemRc,
size: LogicalSize,
_: &CachedRenderingData,
) {
let mut geometry = PhysicalRect::from(size * self.scale_factor);
if geometry.is_empty() {
return;
}
if self.global_alpha_transparent() {
return;
}
let border_color = rect.border_color();
let opaque_border = border_color.is_opaque();
let mut border_width = if border_color.is_transparent() {
PhysicalLength::new(0.)
} else {
rect.border_width() * self.scale_factor
};
let mut fill_radius = rect.border_radius() * self.scale_factor;
fill_radius = fill_radius.outer(border_width / 2. + PhysicalLength::new(1.));
let stroke_border_radius = fill_radius.inner(border_width / 2.);
let (mut background_path, mut maybe_border_path) = if opaque_border {
adjust_rect_and_border_for_inner_drawing(&mut geometry, &mut border_width);
(rect_with_radius_to_path(geometry, stroke_border_radius), None)
} else {
let background_path = rect_with_radius_to_path(geometry, fill_radius);
adjust_rect_and_border_for_inner_drawing(&mut geometry, &mut border_width);
let border_path = rect_with_radius_to_path(geometry, stroke_border_radius);
(background_path, Some(border_path))
};
let fill_paint = self.brush_to_paint(rect.background(), &background_path);
let border_paint = self
.brush_to_paint(
rect.border_color(),
maybe_border_path.as_ref().unwrap_or(&background_path),
)
.map(|mut paint| {
paint.set_line_width(border_width.get());
paint
});
let mut canvas = self.canvas.borrow_mut();
if let Some(paint) = fill_paint {
canvas.fill_path(&background_path, &paint);
}
if let Some(border_paint) = border_paint {
canvas.stroke_path(
maybe_border_path.as_mut().unwrap_or(&mut background_path),
&border_paint,
);
}
}
fn draw_image(
&mut self,
image: Pin<&dyn RenderImage>,
item_rc: &ItemRc,
size: LogicalSize,
_cache: &CachedRenderingData,
) {
self.draw_image_impl(item_rc, image, size);
}
fn draw_text(
&mut self,
text: Pin<&dyn RenderText>,
_: &ItemRc,
size: LogicalSize,
_cache: &CachedRenderingData,
) {
let max_width = size.width_length() * self.scale_factor;
let max_height = size.height_length() * self.scale_factor;
if max_width.get() <= 0. || max_height.get() <= 0. {
return;
}
if self.global_alpha_transparent() {
return;
}
let string = text.text();
let string = string.as_str();
let font = fonts::FONT_CACHE.with(|cache| {
cache.borrow_mut().font(
text.font_request(WindowInner::from_pub(self.window)),
self.scale_factor,
&text.text(),
)
});
let text_path = rect_to_path((size * self.scale_factor).into());
let paint = match self.brush_to_paint(text.color(), &text_path) {
Some(paint) => font.init_paint(text.letter_spacing() * self.scale_factor, paint),
None => return,
};
let (stroke_brush, stroke_width, stroke_style) = text.stroke();
let stroke_width = if stroke_width.get() != 0.0 {
(stroke_width * self.scale_factor).get()
} else {
1.0
};
let stroke_width = match stroke_style {
TextStrokeStyle::Outside => stroke_width * 2.0,
TextStrokeStyle::Center => stroke_width,
};
let stroke_paint = match self.brush_to_paint(stroke_brush.clone(), &text_path) {
Some(mut paint) => {
if stroke_brush.is_transparent() {
None
} else {
paint.set_line_width(stroke_width);
Some(font.init_paint(text.letter_spacing() * self.scale_factor, paint))
}
}
None => None,
};
let mut canvas = self.canvas.borrow_mut();
fonts::layout_text_lines(
string,
&font,
PhysicalSize::from_lengths(max_width, max_height),
text.alignment(),
text.wrap(),
text.overflow(),
false,
None,
&paint,
|to_draw, pos, _, _| {
match (stroke_style, &stroke_paint) {
(TextStrokeStyle::Outside, Some(stroke_paint)) => {
canvas.stroke_text(pos.x, pos.y, to_draw.trim_end(), stroke_paint).unwrap();
canvas.fill_text(pos.x, pos.y, to_draw.trim_end(), &paint).unwrap();
}
(TextStrokeStyle::Center, Some(stroke_paint)) => {
canvas.fill_text(pos.x, pos.y, to_draw.trim_end(), &paint).unwrap();
canvas.stroke_text(pos.x, pos.y, to_draw.trim_end(), stroke_paint).unwrap();
}
_ => {
canvas.fill_text(pos.x, pos.y, to_draw.trim_end(), &paint).unwrap();
}
};
},
);
}
fn draw_text_input(
&mut self,
text_input: Pin<&items::TextInput>,
_: &ItemRc,
size: LogicalSize,
) {
let width = size.width_length() * self.scale_factor;
let height = size.height_length() * self.scale_factor;
if width.get() <= 0. || height.get() <= 0. {
return;
}
if self.global_alpha_transparent() {
return;
}
let font = fonts::FONT_CACHE.with(|cache| {
cache.borrow_mut().font(
text_input.font_request(&WindowInner::from_pub(self.window).window_adapter()),
self.scale_factor,
&text_input.text(),
)
});
let visual_representation = text_input.visual_representation(None);
let paint = match self.brush_to_paint(
visual_representation.text_color,
&rect_to_path((size * self.scale_factor).into()),
) {
Some(paint) => font.init_paint(text_input.letter_spacing() * self.scale_factor, paint),
None => return,
};
let (min_select, max_select) = if !visual_representation.preedit_range.is_empty() {
(visual_representation.preedit_range.start, visual_representation.preedit_range.end)
} else {
(visual_representation.selection_range.start, visual_representation.selection_range.end)
};
let (cursor_visible, cursor_pos) =
if let Some(cursor_pos) = visual_representation.cursor_position {
(true, cursor_pos)
} else {
(false, 0)
};
let mut canvas = self.canvas.borrow_mut();
let font_height = font.height();
let text: SharedString = visual_representation.text.into();
let cursor_point = fonts::layout_text_lines(
text.as_str(),
&font,
PhysicalSize::from_lengths(width, height),
(text_input.horizontal_alignment(), text_input.vertical_alignment()),
text_input.wrap(),
items::TextOverflow::Clip,
text_input.single_line(),
cursor_visible.then_some(cursor_pos),
&paint,
|to_draw: &str, pos: PhysicalPoint, start, metrics: &femtovg::TextMetrics| {
let range = start..(start + to_draw.len());
if min_select != max_select
&& (range.contains(&min_select)
|| range.contains(&max_select)
|| (min_select..max_select).contains(&start))
{
let mut selection_start_x = PhysicalLength::default();
let mut selection_end_x = PhysicalLength::default();
let mut after_selection_x = PhysicalLength::default();
for glyph in &metrics.glyphs {
if glyph.byte_index == min_select.saturating_sub(start) {
selection_start_x = PhysicalLength::new(glyph.x - glyph.bearing_x);
}
if glyph.byte_index == max_select - start
|| glyph.byte_index >= to_draw.len()
{
after_selection_x = PhysicalLength::new(glyph.x - glyph.bearing_x);
break;
}
selection_end_x = PhysicalLength::new(glyph.x + glyph.advance_x);
}
let selection_rect = PhysicalRect::new(
pos + PhysicalPoint::from_lengths(
selection_start_x,
PhysicalLength::default(),
)
.to_vector(),
PhysicalSize::from_lengths(
selection_end_x - selection_start_x,
font_height,
),
);
canvas.fill_path(
&rect_to_path(selection_rect),
&femtovg::Paint::color(to_femtovg_color(
&text_input.selection_background_color(),
)),
);
let mut selected_paint = paint.clone();
selected_paint
.set_color(to_femtovg_color(&text_input.selection_foreground_color()));
canvas
.fill_text(
pos.x,
pos.y,
to_draw[..min_select.saturating_sub(start)].trim_end(),
&paint,
)
.unwrap();
canvas
.fill_text(
pos.x + selection_start_x.get(),
pos.y,
to_draw[min_select.saturating_sub(start)
..(max_select - start).min(to_draw.len())]
.trim_end(),
&selected_paint,
)
.unwrap();
canvas
.fill_text(
pos.x + after_selection_x.get(),
pos.y,
to_draw[(max_select - start).min(to_draw.len())..].trim_end(),
&paint,
)
.unwrap();
} else {
canvas.fill_text(pos.x, pos.y, to_draw.trim_end(), &paint).unwrap();
};
},
);
if let Some(cursor_point) = cursor_point {
let mut cursor_rect = femtovg::Path::new();
cursor_rect.rect(
cursor_point.x,
cursor_point.y,
(text_input.text_cursor_width() * self.scale_factor).get(),
font_height.get(),
);
canvas.fill_path(
&cursor_rect,
&femtovg::Paint::color(to_femtovg_color(&visual_representation.cursor_color)),
);
}
}
fn draw_path(&mut self, path: Pin<&items::Path>, item_rc: &ItemRc, _size: LogicalSize) {
if self.global_alpha_transparent() {
return;
}
let (offset, path_events) = match path.fitted_path_events(item_rc) {
Some(offset_and_events) => offset_and_events,
None => return,
};
let mut femtovg_path = femtovg::Path::new();
#[derive(Default)]
struct OrientationCalculator {
area: f32,
prev: Point,
}
impl OrientationCalculator {
fn add_point(&mut self, p: Point) {
self.area += (p.x - self.prev.x) * (p.y + self.prev.y);
self.prev = p;
}
}
use femtovg::Solidity;
let mut orient = OrientationCalculator::default();
for x in path_events.iter() {
match x {
lyon_path::Event::Begin { at } => {
femtovg_path.solidity(if orient.area < 0. {
Solidity::Hole
} else {
Solidity::Solid
});
femtovg_path
.move_to(at.x * self.scale_factor.get(), at.y * self.scale_factor.get());
orient.area = 0.;
orient.prev = at;
}
lyon_path::Event::Line { from: _, to } => {
femtovg_path
.line_to(to.x * self.scale_factor.get(), to.y * self.scale_factor.get());
orient.add_point(to);
}
lyon_path::Event::Quadratic { from: _, ctrl, to } => {
femtovg_path.quad_to(
ctrl.x * self.scale_factor.get(),
ctrl.y * self.scale_factor.get(),
to.x * self.scale_factor.get(),
to.y * self.scale_factor.get(),
);
orient.add_point(to);
}
lyon_path::Event::Cubic { from: _, ctrl1, ctrl2, to } => {
femtovg_path.bezier_to(
ctrl1.x * self.scale_factor.get(),
ctrl1.y * self.scale_factor.get(),
ctrl2.x * self.scale_factor.get(),
ctrl2.y * self.scale_factor.get(),
to.x * self.scale_factor.get(),
to.y * self.scale_factor.get(),
);
orient.add_point(to);
}
lyon_path::Event::End { last: _, first: _, close } => {
femtovg_path.solidity(if orient.area < 0. {
Solidity::Hole
} else {
Solidity::Solid
});
if close {
femtovg_path.close()
}
}
}
}
let fill_paint = self.brush_to_paint(path.fill(), &femtovg_path).map(|mut fill_paint| {
fill_paint.set_fill_rule(match path.fill_rule() {
FillRule::Nonzero => femtovg::FillRule::NonZero,
FillRule::Evenodd => femtovg::FillRule::EvenOdd,
});
fill_paint
});
let border_paint = self.brush_to_paint(path.stroke(), &femtovg_path).map(|mut paint| {
paint.set_line_width((path.stroke_width() * self.scale_factor).get());
paint
});
self.canvas.borrow_mut().save_with(|canvas| {
canvas.translate(offset.x, offset.y);
if let Some(fill_paint) = &fill_paint {
canvas.fill_path(&femtovg_path, fill_paint);
}
if let Some(border_paint) = &border_paint {
canvas.stroke_path(&femtovg_path, border_paint);
}
})
}
fn draw_box_shadow(
&mut self,
box_shadow: Pin<&items::BoxShadow>,
item_rc: &ItemRc,
_size: LogicalSize,
) {
if box_shadow.color().alpha() == 0
|| (box_shadow.blur() == LogicalLength::zero()
&& box_shadow.offset_x() == LogicalLength::zero()
&& box_shadow.offset_y() == LogicalLength::zero())
{
return;
}
let cache_entry = self.box_shadow_cache.get_box_shadow(
item_rc,
self.graphics_cache,
box_shadow,
self.scale_factor,
|shadow_options| {
let blur = shadow_options.blur;
let width = shadow_options.width;
let height = shadow_options.height;
let radius = shadow_options.radius;
let shadow_rect = PhysicalRect::new(
PhysicalPoint::default(),
PhysicalSize::from_lengths(width + blur * 2., height + blur * 2.),
);
let shadow_image_width = shadow_rect.width().ceil() as u32;
let shadow_image_height = shadow_rect.height().ceil() as u32;
let shadow_image = Texture::new_empty_on_gpu(
&self.canvas,
shadow_image_width,
shadow_image_height,
)
.expect("unable to create box shadow texture");
{
let mut canvas = self.canvas.borrow_mut();
canvas.save();
canvas.set_render_target(shadow_image.as_render_target());
canvas.reset();
canvas.clear_rect(
0,
0,
shadow_rect.width().ceil() as u32,
shadow_rect.height().ceil() as u32,
femtovg::Color::rgba(0, 0, 0, 0),
);
let mut shadow_path = femtovg::Path::new();
shadow_path.rounded_rect(
blur.get(),
blur.get(),
width.get(),
height.get(),
radius.get(),
);
canvas.fill_path(
&shadow_path,
&femtovg::Paint::color(femtovg::Color::rgb(255, 255, 255)),
);
}
let shadow_image = if blur.get() > 0. {
let blurred_image = shadow_image
.filter(femtovg::ImageFilter::GaussianBlur { sigma: blur.get() / 2. });
self.canvas.borrow_mut().set_render_target(blurred_image.as_render_target());
self.textures_to_delete_after_flush.borrow_mut().push(shadow_image);
blurred_image
} else {
shadow_image
};
{
let mut canvas = self.canvas.borrow_mut();
canvas.global_composite_operation(femtovg::CompositeOperation::SourceIn);
let mut shadow_image_rect = femtovg::Path::new();
shadow_image_rect.rect(0., 0., shadow_rect.width(), shadow_rect.height());
canvas.fill_path(
&shadow_image_rect,
&femtovg::Paint::color(to_femtovg_color(&box_shadow.color())),
);
canvas.restore();
canvas.set_render_target(self.current_render_target());
}
ItemGraphicsCacheEntry::Texture(shadow_image)
},
);
let shadow_image = match &cache_entry {
Some(cached_shadow_image) => cached_shadow_image.as_texture(),
None => return, };
let shadow_image_size = match shadow_image.size() {
Some(size) => size,
None => return,
};
let shadow_image_paint = shadow_image.as_paint().with_anti_alias(false);
let mut shadow_image_rect = femtovg::Path::new();
shadow_image_rect.rect(
0.,
0.,
shadow_image_size.width as f32,
shadow_image_size.height as f32,
);
self.canvas.borrow_mut().save_with(|canvas| {
let blur = box_shadow.blur() * self.scale_factor;
let offset = LogicalPoint::from_lengths(box_shadow.offset_x(), box_shadow.offset_y())
* self.scale_factor;
canvas.translate(offset.x - blur.get(), offset.y - blur.get());
canvas.fill_path(&shadow_image_rect, &shadow_image_paint);
});
}
fn visit_opacity(
&mut self,
opacity_item: Pin<&Opacity>,
item_rc: &ItemRc,
_size: LogicalSize,
) -> RenderingResult {
let opacity = opacity_item.opacity();
if Opacity::need_layer(item_rc, opacity) {
self.render_and_blend_layer(opacity, item_rc)
} else {
self.apply_opacity(opacity);
self.graphics_cache.release(item_rc);
RenderingResult::ContinueRenderingChildren
}
}
fn visit_layer(
&mut self,
layer_item: Pin<&Layer>,
self_rc: &ItemRc,
_size: LogicalSize,
) -> RenderingResult {
if layer_item.cache_rendering_hint() {
self.render_and_blend_layer(1.0, self_rc)
} else {
self.graphics_cache.release(self_rc);
RenderingResult::ContinueRenderingChildren
}
}
fn visit_clip(
&mut self,
clip_item: Pin<&Clip>,
item_rc: &ItemRc,
size: LogicalSize,
) -> RenderingResult {
if !clip_item.clip() {
return RenderingResult::ContinueRenderingChildren;
}
let geometry = LogicalRect::from(size);
if !self.get_current_clip().intersects(&geometry) {
return RenderingResult::ContinueRenderingWithoutChildren;
}
let radius = clip_item.logical_border_radius();
let border_width = clip_item.border_width();
if !radius.is_zero() {
if let Some(layer_image) = self.render_layer(item_rc, &|| item_rc.geometry().size) {
let layer_image_paint = layer_image.as_paint();
let layer_path = clip_path_for_rect_alike_item(
geometry,
radius,
border_width,
self.scale_factor,
);
self.canvas.borrow_mut().fill_path(&layer_path, &layer_image_paint);
}
RenderingResult::ContinueRenderingWithoutChildren
} else {
self.graphics_cache.release(item_rc);
self.combine_clip(geometry, radius, border_width);
RenderingResult::ContinueRenderingChildren
}
}
fn combine_clip(
&mut self,
clip_rect: LogicalRect,
radius: LogicalBorderRadius,
border_width: LogicalLength,
) -> bool {
let clip = &mut self.state.last_mut().unwrap().scissor;
let clip_region_valid = match clip.intersection(&clip_rect) {
Some(r) => {
*clip = r;
true
}
None => {
*clip = LogicalRect::default();
false
}
};
let clip_path =
clip_path_for_rect_alike_item(clip_rect, radius, border_width, self.scale_factor);
let clip_path_bounds = path_bounding_box(&self.canvas, &clip_path);
self.canvas.borrow_mut().intersect_scissor(
clip_path_bounds.min.x,
clip_path_bounds.min.y,
clip_path_bounds.width(),
clip_path_bounds.height(),
);
debug_assert!(radius.is_zero());
clip_region_valid
}
fn get_current_clip(&self) -> LogicalRect {
self.state.last().unwrap().scissor
}
fn save_state(&mut self) {
self.canvas.borrow_mut().save();
self.state.push(self.state.last().unwrap().clone());
}
fn restore_state(&mut self) {
self.state.pop();
self.canvas.borrow_mut().restore();
}
fn scale_factor(&self) -> f32 {
self.scale_factor.get()
}
fn draw_cached_pixmap(
&mut self,
item_rc: &ItemRc,
update_fn: &dyn Fn(&mut dyn FnMut(u32, u32, &[u8])),
) {
let canvas = &self.canvas;
let cache_entry = self.graphics_cache.get_or_update_cache_entry(item_rc, || {
let mut cached_image = None;
update_fn(&mut |width: u32, height: u32, data: &[u8]| {
use rgb::FromSlice;
let img = imgref::Img::new(data.as_rgba(), width as usize, height as usize);
if let Ok(image_id) =
canvas.borrow_mut().create_image(img, femtovg::ImageFlags::PREMULTIPLIED)
{
cached_image =
Some(ItemGraphicsCacheEntry::Texture(Texture::adopt(canvas, image_id)))
};
});
cached_image
});
let image_id = match cache_entry {
Some(ItemGraphicsCacheEntry::Texture(image)) => image.id,
Some(ItemGraphicsCacheEntry::ColorizedImage { .. }) => unreachable!(),
None => return,
};
let mut canvas = self.canvas.borrow_mut();
let image_info = canvas.image_info(image_id).unwrap();
let (width, height) = (image_info.width() as f32, image_info.height() as f32);
let fill_paint = femtovg::Paint::image(image_id, 0., 0., width, height, 0.0, 1.0);
let mut path = femtovg::Path::new();
path.rect(0., 0., width, height);
canvas.fill_path(&path, &fill_paint);
}
fn draw_string(&mut self, string: &str, color: Color) {
let font = fonts::FONT_CACHE
.with(|cache| cache.borrow_mut().font(Default::default(), self.scale_factor, string));
let paint = font
.init_paint(PhysicalLength::default(), femtovg::Paint::color(to_femtovg_color(&color)));
let mut canvas = self.canvas.borrow_mut();
canvas.fill_text(0., 0., string, &paint).unwrap();
}
fn draw_image_direct(&mut self, image: i_slint_core::graphics::Image) {
let image_size = image.size();
let target_width = LogicalLength::new(image_size.width as _);
let target_height = LogicalLength::new(image_size.height as _);
let target_w = target_width * self.scale_factor;
let target_h = target_height * self.scale_factor;
if target_w.get() <= 0. || target_h.get() <= 0. {
return;
}
let image_inner: &ImageInner = (&image).into();
let Some(cached_image) = TextureCacheKey::new(image_inner, None, Default::default())
.and_then(|cache_key| {
self.texture_cache.borrow_mut().lookup_image_in_cache_or_create(cache_key, || {
Texture::new_from_image(image_inner, &self.canvas, None, Default::default())
})
})
.or_else(|| {
Texture::new_from_image(image_inner, &self.canvas, None, Default::default())
})
else {
return;
};
let image_id = cached_image.id;
let image_size = cached_image.size().unwrap_or_default().cast();
let (source_width, source_height) = (image_size.width, image_size.height);
let fill_paint =
femtovg::Paint::image(image_id, 0., 0., image_size.width, image_size.height, 0.0, 1.0)
.with_anti_alias(false);
let mut path = femtovg::Path::new();
path.rect(0., 0., source_width, source_height);
self.canvas.borrow_mut().save_with(|canvas| {
canvas.fill_path(&path, &fill_paint);
})
}
fn window(&self) -> &i_slint_core::window::WindowInner {
i_slint_core::window::WindowInner::from_pub(self.window)
}
fn as_any(&mut self) -> Option<&mut dyn std::any::Any> {
None
}
fn translate(&mut self, distance: LogicalVector) {
let physical_distance = distance * self.scale_factor;
self.canvas.borrow_mut().translate(physical_distance.x, physical_distance.y);
let clip = &mut self.state.last_mut().unwrap().scissor;
*clip = clip.translate(-distance)
}
fn rotate(&mut self, angle_in_degrees: f32) {
let angle_in_radians = angle_in_degrees.to_radians();
self.canvas.borrow_mut().rotate(angle_in_radians);
let clip = &mut self.state.last_mut().unwrap().scissor;
let (sin, cos) = angle_in_radians.sin_cos();
let rotate_point = |p: LogicalPoint| (p.x * cos - p.y * sin, p.x * sin + p.y * cos);
let corners = [
rotate_point(clip.origin),
rotate_point(clip.origin + euclid::vec2(clip.width(), 0.)),
rotate_point(clip.origin + euclid::vec2(0., clip.height())),
rotate_point(clip.origin + clip.size),
];
let origin: LogicalPoint = (
corners.iter().fold(f32::MAX, |a, b| b.0.min(a)),
corners.iter().fold(f32::MAX, |a, b| b.1.min(a)),
)
.into();
let end: LogicalPoint = (
corners.iter().fold(f32::MIN, |a, b| b.0.max(a)),
corners.iter().fold(f32::MIN, |a, b| b.1.max(a)),
)
.into();
*clip = LogicalRect::new(origin, (end - origin).into());
}
fn apply_opacity(&mut self, opacity: f32) {
let state = &mut self.state.last_mut().unwrap().global_alpha;
*state *= opacity;
self.canvas.borrow_mut().set_global_alpha(*state);
}
fn metrics(&self) -> RenderingMetrics {
self.metrics.clone()
}
}
impl<'a> GLItemRenderer<'a> {
pub(super) fn new(
canvas: &CanvasRc,
graphics_cache: &'a ItemGraphicsCache,
texture_cache: &'a RefCell<super::images::TextureCache>,
window: &'a i_slint_core::api::Window,
width: u32,
height: u32,
) -> Self {
let scale_factor = ScaleFactor::new(window.scale_factor());
Self {
graphics_cache,
texture_cache,
box_shadow_cache: Default::default(),
canvas: canvas.clone(),
textures_to_delete_after_flush: Default::default(),
window,
scale_factor,
state: vec![State {
scissor: LogicalRect::new(
LogicalPoint::default(),
PhysicalSize::new(width as f32, height as f32) / scale_factor,
),
global_alpha: 1.,
current_render_target: femtovg::RenderTarget::Screen,
}],
metrics: RenderingMetrics { layers_created: Some(0) },
}
}
fn render_layer(
&mut self,
item_rc: &ItemRc,
layer_logical_size_fn: &dyn Fn() -> LogicalSize,
) -> Option<Rc<Texture>> {
let existing_layer_texture =
self.graphics_cache.with_entry(item_rc, |cache_entry| match cache_entry {
Some(ItemGraphicsCacheEntry::Texture(texture)) => Some(texture.clone()),
_ => None,
});
let cache_entry = self.graphics_cache.get_or_update_cache_entry(item_rc, || {
ItemGraphicsCacheEntry::Texture({
let size = (layer_logical_size_fn() * self.scale_factor).ceil().try_cast()?;
let layer_image = existing_layer_texture
.and_then(|layer_texture| {
debug_assert_eq!(Rc::strong_count(&layer_texture), 2);
if layer_texture.size() == Some(size.to_untyped()) {
Some(layer_texture)
} else {
None
}
})
.or_else(|| {
*self.metrics.layers_created.as_mut().unwrap() += 1;
Texture::new_empty_on_gpu(&self.canvas, size.width, size.height)
})?;
let previous_render_target = self.current_render_target();
{
let mut canvas = self.canvas.borrow_mut();
canvas.save();
canvas.set_render_target(layer_image.as_render_target());
canvas.reset();
canvas.clear_rect(
0,
0,
size.width,
size.height,
femtovg::Color::rgba(0, 0, 0, 0),
);
}
*self.state.last_mut().unwrap() = State {
scissor: LogicalRect::new(
LogicalPoint::default(),
PhysicalSize::new(size.width as f32, size.height as f32)
/ self.scale_factor,
),
global_alpha: 1.,
current_render_target: layer_image.as_render_target(),
};
i_slint_core::item_rendering::render_item_children(
self,
item_rc.item_tree(),
item_rc.index() as isize,
);
{
let mut canvas = self.canvas.borrow_mut();
canvas.restore();
canvas.set_render_target(previous_render_target);
}
layer_image
})
.into()
});
cache_entry.map(|item_cache_entry| item_cache_entry.as_texture().clone())
}
fn render_and_blend_layer(&mut self, alpha_tint: f32, item_rc: &ItemRc) -> RenderingResult {
let current_clip = self.get_current_clip();
if let Some((layer_image, layer_size)) = self
.render_layer(item_rc, &|| {
let children_rect = i_slint_core::properties::evaluate_no_tracking(|| {
item_rc.geometry().union(
&i_slint_core::item_rendering::item_children_bounding_rect(
item_rc.item_tree(),
item_rc.index() as isize,
¤t_clip,
),
)
});
children_rect.size
})
.and_then(|image| image.size().map(|size| (image, size)))
{
let mut layer_path = femtovg::Path::new();
let layer_image_paint =
layer_image.as_paint_with_alpha(alpha_tint).with_anti_alias(false);
layer_path.rect(0., 0., layer_size.width as _, layer_size.height as _);
self.canvas.borrow_mut().fill_path(&layer_path, &layer_image_paint);
}
RenderingResult::ContinueRenderingWithoutChildren
}
fn colorize_image(
&self,
original_cache_entry: ItemGraphicsCacheEntry,
colorize_brush: Brush,
scaling: ImageRendering,
) -> ItemGraphicsCacheEntry {
if colorize_brush.is_transparent() {
return original_cache_entry;
};
let original_image = original_cache_entry.as_texture();
let image_size = match original_image.size() {
Some(size) => size,
None => return original_cache_entry,
};
let scaling_flags = super::images::base_image_flags(scaling);
let image_id = original_image.id;
let colorized_image = self
.canvas
.borrow_mut()
.create_image_empty(
image_size.width as usize,
image_size.height as usize,
femtovg::PixelFormat::Rgba8,
femtovg::ImageFlags::PREMULTIPLIED | scaling_flags,
)
.expect("internal error allocating temporary texture for image colorization");
let image_size: Size = image_size.cast();
let mut image_rect = femtovg::Path::new();
image_rect.rect(0., 0., image_size.width, image_size.height);
let brush_paint = match self.brush_to_paint(colorize_brush, &image_rect) {
Some(paint) => paint.with_anti_alias(false),
None => return original_cache_entry,
};
self.canvas.borrow_mut().save_with(|canvas| {
canvas.reset();
canvas.scale(1., -1.); canvas.translate(0., -image_size.height);
canvas.set_render_target(femtovg::RenderTarget::Image(colorized_image));
canvas.global_composite_operation(femtovg::CompositeOperation::Copy);
canvas.fill_path(
&image_rect,
&femtovg::Paint::image(
image_id,
0.,
0.,
image_size.width,
image_size.height,
0.,
1.0,
),
);
canvas.global_composite_operation(femtovg::CompositeOperation::SourceIn);
canvas.fill_path(&image_rect, &brush_paint);
canvas.set_render_target(self.current_render_target());
});
ItemGraphicsCacheEntry::ColorizedImage {
_original_image: original_image.clone(),
colorized_image: Texture::adopt(&self.canvas, colorized_image),
}
}
fn draw_image_impl(
&mut self,
item_rc: &ItemRc,
item: Pin<&dyn RenderImage>,
size: LogicalSize,
) {
if size.width <= 0. || size.height <= 0. {
return;
}
let cached_image = loop {
let image_cache_entry = self.graphics_cache.get_or_update_cache_entry(item_rc, || {
let image = item.source();
let image_inner: &ImageInner = (&image).into();
let target_size_for_scalable_source = if image_inner.is_svg() {
let image_size = image.size().cast::<f32>();
if image_size.is_empty() {
return None;
}
let t = item.target_size() * self.scale_factor;
let fit = i_slint_core::graphics::fit(
item.image_fit(),
t,
IntRect::from_size(image_size.cast()),
self.scale_factor,
Default::default(), item.tiling(),
);
Some(euclid::size2(
(image_size.width * fit.source_to_target_x) as u32,
(image_size.height * fit.source_to_target_y) as u32,
))
} else {
None
};
let image_rendering = item.rendering();
TextureCacheKey::new(image_inner, target_size_for_scalable_source, image_rendering)
.and_then(|cache_key| {
self.texture_cache.borrow_mut().lookup_image_in_cache_or_create(
cache_key,
|| {
Texture::new_from_image(
image_inner,
&self.canvas,
target_size_for_scalable_source,
image_rendering,
)
},
)
})
.or_else(|| {
Texture::new_from_image(
image_inner,
&self.canvas,
target_size_for_scalable_source,
image_rendering,
)
})
.map(ItemGraphicsCacheEntry::Texture)
.map(|cache_entry| {
self.colorize_image(cache_entry, item.colorize(), image_rendering)
})
});
let cached_image = match image_cache_entry {
Some(entry) if entry.as_texture().size().is_some() => entry,
_ => {
return;
}
};
if !cached_image.is_colorized_image() && !item.colorize().is_transparent() {
self.graphics_cache.release(item_rc);
continue;
}
break cached_image.as_texture().clone();
};
let image = item.source();
let image_id = cached_image.id;
let orig_size = image.size().cast::<f32>();
let buf_size = cached_image.size().unwrap_or_default().cast::<f32>();
let source_clip_rect = item.source_clip().unwrap_or(IntRect::from_size(orig_size.cast()));
let image_inner: &ImageInner = (&image).into();
let fits = if let ImageInner::NineSlice(nine) = image_inner {
i_slint_core::graphics::fit9slice(
image.size(),
nine.1,
size * self.scale_factor,
self.scale_factor,
item.alignment(),
item.tiling(),
)
.collect::<Vec<_>>()
} else {
vec![i_slint_core::graphics::fit(
item.image_fit(),
size * self.scale_factor,
source_clip_rect,
self.scale_factor,
item.alignment(),
item.tiling(),
)]
};
let scale_w = buf_size.width / orig_size.width;
let scale_h = buf_size.height / orig_size.height;
for fit in fits {
let (image_id, origin, texture_size) =
if fit.tiled.is_some() && fit.clip_rect.size.cast() != orig_size {
let scaling_flags = match item.rendering() {
ImageRendering::Smooth => femtovg::ImageFlags::empty(),
ImageRendering::Pixelated => {
femtovg::ImageFlags::empty() | femtovg::ImageFlags::NEAREST
}
};
let texture_size = euclid::size2(
scale_w * fit.clip_rect.width() as f32,
scale_h * fit.clip_rect.height() as f32,
);
let clipped_image = self
.canvas
.borrow_mut()
.create_image_empty(
texture_size.width as usize,
texture_size.height as usize,
femtovg::PixelFormat::Rgba8,
femtovg::ImageFlags::PREMULTIPLIED
| femtovg::ImageFlags::REPEAT_X
| femtovg::ImageFlags::REPEAT_Y
| scaling_flags,
)
.expect("internal error allocating temporary texture for image tiling");
let mut image_rect = femtovg::Path::new();
image_rect.rect(0., 0., texture_size.width, texture_size.height);
self.canvas.borrow_mut().save_with(|canvas| {
canvas.reset();
canvas.scale(1., -1.); canvas.translate(0., -scale_h * (fit.clip_rect.height() as f32));
canvas.set_render_target(femtovg::RenderTarget::Image(clipped_image));
canvas.global_composite_operation(femtovg::CompositeOperation::Copy);
canvas.fill_path(
&image_rect,
&femtovg::Paint::image(
image_id,
-scale_w * fit.clip_rect.origin.x as f32,
-scale_h * fit.clip_rect.origin.y as f32,
buf_size.cast().width,
buf_size.cast().height,
0.,
1.0,
),
);
canvas.set_render_target(self.current_render_target());
});
self.textures_to_delete_after_flush
.borrow_mut()
.push(Texture::adopt(&self.canvas, clipped_image));
(clipped_image, Default::default(), texture_size)
} else {
(image_id, fit.clip_rect.origin.cast::<f32>(), buf_size)
};
let tiled = fit.tiled.unwrap_or_default();
let fill_paint = femtovg::Paint::image(
image_id,
-origin.x - tiled.x as f32,
-origin.y - tiled.y as f32,
texture_size.width,
texture_size.height,
0.0,
1.0,
)
.with_anti_alias(fit.source_to_target_x != 1. || fit.source_to_target_y != 0.);
let mut path = femtovg::Path::new();
path.rect(
0.,
0.,
scale_w * fit.size.width / fit.source_to_target_x,
scale_h * fit.size.height / fit.source_to_target_y,
);
self.canvas.borrow_mut().save_with(|canvas| {
canvas.translate(fit.offset.x, fit.offset.y);
canvas.scale(fit.source_to_target_x / scale_w, fit.source_to_target_y / scale_h);
canvas.fill_path(&path, &fill_paint);
})
}
}
fn brush_to_paint(&self, brush: Brush, path: &femtovg::Path) -> Option<femtovg::Paint> {
if brush.is_transparent() {
return None;
}
Some(match brush {
Brush::SolidColor(color) => femtovg::Paint::color(to_femtovg_color(&color)),
Brush::LinearGradient(gradient) => {
let path_bounds = path_bounding_box(&self.canvas, path);
let path_width = path_bounds.width();
let path_height = path_bounds.height();
let (start, end) = i_slint_core::graphics::line_for_angle(
gradient.angle(),
[path_width, path_height].into(),
);
let stops =
gradient.stops().map(|stop| (stop.position, to_femtovg_color(&stop.color)));
femtovg::Paint::linear_gradient_stops(start.x, start.y, end.x, end.y, stops)
}
Brush::RadialGradient(gradient) => {
let path_bounds = path_bounding_box(&self.canvas, path);
let path_width = path_bounds.width();
let path_height = path_bounds.height();
let stops =
gradient.stops().map(|stop| (stop.position, to_femtovg_color(&stop.color)));
femtovg::Paint::radial_gradient_stops(
path_width / 2.,
path_height / 2.,
0.,
(path_width + path_height) / 4.,
stops,
)
}
_ => return None,
})
}
fn current_render_target(&self) -> femtovg::RenderTarget {
self.state.last().unwrap().current_render_target
}
}
pub fn to_femtovg_color(col: &Color) -> femtovg::Color {
femtovg::Color::rgba(col.red(), col.green(), col.blue(), col.alpha())
}