slint/internal/renderers/skia/cached_image.rs
Simon Hausmann 3101104f5c Switch from WGPU 24 to WGPU 25
- Use a version of FemtoVG that uses WGPU 25
- Rename the unstable-wgpu-24 feature and wgpu_24 module
- Fetch bevy from a revision on master branch that switches to WGPU 25
- Move the bevy example out of the common workspace: That way users of the repo don't get bevy pulled from git, unless they want to build the bevy example.
2025-07-09 11:30:05 +02:00

147 lines
5.4 KiB
Rust

// Copyright © SixtyFPS GmbH <info@slint.dev>
// SPDX-License-Identifier: GPL-3.0-only OR LicenseRef-Slint-Royalty-free-2.0 OR LicenseRef-Slint-Software-3.0
use crate::PhysicalSize;
use i_slint_core::graphics::{
cache as core_cache, Image, ImageCacheKey, ImageInner, IntRect, IntSize, OpaqueImage,
OpaqueImageVTable, SharedImageBuffer,
};
use i_slint_core::items::ImageFit;
use i_slint_core::lengths::{LogicalSize, ScaleFactor};
struct SkiaCachedImage {
image: skia_safe::Image,
cache_key: ImageCacheKey,
}
i_slint_core::OpaqueImageVTable_static! {
static SKIA_CACHED_IMAGE_VT for SkiaCachedImage
}
impl OpaqueImage for SkiaCachedImage {
fn size(&self) -> IntSize {
IntSize::new(self.image.width() as u32, self.image.height() as u32)
}
fn cache_key(&self) -> ImageCacheKey {
self.cache_key.clone()
}
}
pub(crate) fn as_skia_image(
image: Image,
target_size_fn: &dyn Fn() -> LogicalSize,
image_fit: ImageFit,
scale_factor: ScaleFactor,
canvas: &skia_safe::Canvas,
surface: Option<&dyn crate::Surface>,
) -> Option<skia_safe::Image> {
let image_inner: &ImageInner = (&image).into();
match image_inner {
ImageInner::None => None,
ImageInner::EmbeddedImage { buffer, cache_key } => {
let result = image_buffer_to_skia_image(buffer);
if let Some(img) = result.as_ref() {
core_cache::replace_cached_image(
cache_key.clone(),
ImageInner::BackendStorage(vtable::VRc::into_dyn(vtable::VRc::new(
SkiaCachedImage { image: img.clone(), cache_key: cache_key.clone() },
))),
)
}
result
}
ImageInner::Svg(svg) => {
// Query target_width/height here again to ensure that changes will invalidate the item rendering cache.
let svg_size = svg.size();
let fit = i_slint_core::graphics::fit(
image_fit,
target_size_fn() * scale_factor,
IntRect::from_size(svg_size.cast()),
scale_factor,
Default::default(), // We only care about the size, so alignments don't matter
Default::default(),
);
let target_size = PhysicalSize::new(
svg_size.cast::<f32>().width * fit.source_to_target_x,
svg_size.cast::<f32>().height * fit.source_to_target_y,
);
let pixels = match svg.render(Some(target_size.cast())).ok()? {
SharedImageBuffer::RGB8(_) => unreachable!(),
SharedImageBuffer::RGBA8(_) => unreachable!(),
SharedImageBuffer::RGBA8Premultiplied(pixels) => pixels,
};
let image_info = skia_safe::ImageInfo::new(
skia_safe::ISize::new(pixels.width() as i32, pixels.height() as i32),
skia_safe::ColorType::RGBA8888,
skia_safe::AlphaType::Premul,
None,
);
skia_safe::images::raster_from_data(
&image_info,
skia_safe::Data::new_copy(pixels.as_bytes()),
pixels.width() as usize * 4,
)
}
ImageInner::StaticTextures(_) => todo!(),
ImageInner::BackendStorage(x) => {
vtable::VRc::borrow(x).downcast::<SkiaCachedImage>().map(|x| x.image.clone())
}
ImageInner::BorrowedOpenGLTexture(texture) => {
surface.and_then(|surface| surface.import_opengl_texture(canvas, texture))
}
ImageInner::NineSlice(n) => as_skia_image(
n.image(),
target_size_fn,
ImageFit::Preserve,
scale_factor,
canvas,
surface,
),
#[cfg(feature = "unstable-wgpu-25")]
ImageInner::WGPUTexture(..) => None,
}
}
fn image_buffer_to_skia_image(buffer: &SharedImageBuffer) -> Option<skia_safe::Image> {
let (data, bpl, size, color_type, alpha_type) = match buffer {
SharedImageBuffer::RGB8(pixels) => {
// RGB888 with one byte per component is not supported by Skia right now. Convert once to RGBA8 :-(
let rgba = pixels
.as_bytes()
.chunks(3)
.flat_map(|rgb| IntoIterator::into_iter([rgb[0], rgb[1], rgb[2], 255]))
.collect::<Vec<u8>>();
(
skia_safe::Data::new_copy(&*rgba),
pixels.width() as usize * 4,
pixels.size(),
skia_safe::ColorType::RGBA8888,
skia_safe::AlphaType::Unpremul,
)
}
SharedImageBuffer::RGBA8(pixels) => (
skia_safe::Data::new_copy(pixels.as_bytes()),
pixels.width() as usize * 4,
pixels.size(),
skia_safe::ColorType::RGBA8888,
skia_safe::AlphaType::Unpremul,
),
SharedImageBuffer::RGBA8Premultiplied(pixels) => (
skia_safe::Data::new_copy(pixels.as_bytes()),
pixels.width() as usize * 4,
pixels.size(),
skia_safe::ColorType::RGBA8888,
skia_safe::AlphaType::Premul,
),
};
let image_info = skia_safe::ImageInfo::new(
skia_safe::ISize::new(size.width as i32, size.height as i32),
color_type,
alpha_type,
None,
);
skia_safe::images::raster_from_data(&image_info, data, bpl)
}