mirror of
https://github.com/slint-ui/slint.git
synced 2025-07-08 21:55:28 +00:00

To be compatible with the 2024 edition, we need to wrap the `no_mangle` attribute in `unsafe()`. The parsing for that in cbindgen was only added in the version 0.28, but we couldn't upgrade cbindgen before because of a regression in cbindgen 0.27 that prevented us from upgrading. Now that cbindgen 0.29 is released with a fix, we can prepare for the 2024 edition
906 lines
32 KiB
Rust
906 lines
32 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 alloc::rc::Rc;
|
|
use alloc::{boxed::Box, string::String};
|
|
use core::ffi::c_void;
|
|
use i_slint_core::api::{
|
|
LogicalSize, PhysicalPosition, PhysicalSize, Window, WindowPosition, WindowSize,
|
|
};
|
|
use i_slint_core::graphics::euclid;
|
|
use i_slint_core::graphics::IntSize;
|
|
use i_slint_core::platform::{Clipboard, Platform, PlatformError};
|
|
use i_slint_core::renderer::Renderer;
|
|
use i_slint_core::window::ffi::WindowAdapterRcOpaque;
|
|
use i_slint_core::window::{WindowAdapter, WindowProperties};
|
|
use i_slint_core::{Brush, SharedString};
|
|
|
|
type WindowAdapterUserData = *mut c_void;
|
|
|
|
// FIXME wrapper over &dyn Renderer
|
|
#[repr(C)]
|
|
pub struct RendererPtr {
|
|
_a: *const c_void,
|
|
_b: *const c_void,
|
|
}
|
|
|
|
pub struct CppWindowAdapter {
|
|
window: Window,
|
|
user_data: WindowAdapterUserData,
|
|
drop: unsafe extern "C" fn(WindowAdapterUserData),
|
|
/// Safety: the returned pointer must live for the lifetime of self
|
|
get_renderer_ref: unsafe extern "C" fn(WindowAdapterUserData) -> RendererPtr,
|
|
set_visible: unsafe extern "C" fn(WindowAdapterUserData, bool),
|
|
request_redraw: unsafe extern "C" fn(WindowAdapterUserData),
|
|
size: unsafe extern "C" fn(WindowAdapterUserData) -> IntSize,
|
|
set_size: unsafe extern "C" fn(WindowAdapterUserData, IntSize),
|
|
update_window_properties: unsafe extern "C" fn(WindowAdapterUserData, &WindowProperties),
|
|
position:
|
|
unsafe extern "C" fn(WindowAdapterUserData, &mut euclid::default::Point2D<i32>) -> bool,
|
|
set_position: unsafe extern "C" fn(WindowAdapterUserData, euclid::default::Point2D<i32>),
|
|
}
|
|
|
|
impl Drop for CppWindowAdapter {
|
|
fn drop(&mut self) {
|
|
unsafe { (self.drop)(self.user_data) };
|
|
}
|
|
}
|
|
|
|
impl WindowAdapter for CppWindowAdapter {
|
|
fn window(&self) -> &Window {
|
|
&self.window
|
|
}
|
|
|
|
fn set_visible(&self, visible: bool) -> Result<(), PlatformError> {
|
|
unsafe { (self.set_visible)(self.user_data, visible) };
|
|
Ok(())
|
|
}
|
|
|
|
fn position(&self) -> Option<PhysicalPosition> {
|
|
let mut pos = euclid::default::Point2D::<i32>::default();
|
|
if unsafe { (self.position)(self.user_data, &mut pos) } {
|
|
Some(i_slint_core::graphics::ffi::physical_position_to_api(pos))
|
|
} else {
|
|
None
|
|
}
|
|
}
|
|
|
|
fn set_position(&self, position: WindowPosition) {
|
|
let physical_position = i_slint_core::graphics::ffi::physical_position_from_api(
|
|
position.to_physical(self.window.scale_factor()),
|
|
);
|
|
unsafe { (self.set_position)(self.user_data, physical_position) }
|
|
}
|
|
|
|
fn set_size(&self, size: WindowSize) {
|
|
let physical_size = i_slint_core::graphics::ffi::physical_size_from_api(
|
|
size.to_physical(self.window.scale_factor()),
|
|
);
|
|
unsafe { (self.set_size)(self.user_data, physical_size) }
|
|
}
|
|
|
|
fn size(&self) -> PhysicalSize {
|
|
let s = unsafe { (self.size)(self.user_data) };
|
|
PhysicalSize::new(s.width, s.height)
|
|
}
|
|
|
|
fn renderer(&self) -> &dyn Renderer {
|
|
unsafe { core::mem::transmute((self.get_renderer_ref)(self.user_data)) }
|
|
}
|
|
|
|
fn request_redraw(&self) {
|
|
unsafe { (self.request_redraw)(self.user_data) }
|
|
}
|
|
|
|
fn update_window_properties(&self, properties: WindowProperties<'_>) {
|
|
unsafe { (self.update_window_properties)(self.user_data, &properties) }
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_window_properties_get_title(wp: &WindowProperties, out: &mut SharedString) {
|
|
*out = wp.title();
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_window_properties_get_background(wp: &WindowProperties, out: &mut Brush) {
|
|
*out = wp.background();
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_window_properties_get_fullscreen(wp: &WindowProperties) -> bool {
|
|
wp.is_fullscreen()
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_window_properties_get_minimized(wp: &WindowProperties) -> bool {
|
|
wp.is_minimized()
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_window_properties_get_maximized(wp: &WindowProperties) -> bool {
|
|
wp.is_maximized()
|
|
}
|
|
|
|
#[repr(C)]
|
|
#[derive(Clone, Copy)]
|
|
/// a Repr(C) variant of slint::platform::LayoutConstraints
|
|
pub struct LayoutConstraintsReprC {
|
|
pub min: i_slint_core::graphics::Size,
|
|
pub max: i_slint_core::graphics::Size,
|
|
pub preferred: i_slint_core::graphics::Size,
|
|
pub has_min: bool,
|
|
pub has_max: bool,
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_window_properties_get_layout_constraints(
|
|
wp: &WindowProperties,
|
|
) -> LayoutConstraintsReprC {
|
|
let c = wp.layout_constraints();
|
|
LayoutConstraintsReprC {
|
|
min: i_slint_core::lengths::logical_size_from_api(c.min.unwrap_or_default()).to_untyped(),
|
|
max: i_slint_core::lengths::logical_size_from_api(
|
|
c.max.unwrap_or(LogicalSize { width: f32::MAX, height: f32::MAX }),
|
|
)
|
|
.to_untyped(),
|
|
preferred: i_slint_core::lengths::logical_size_from_api(c.preferred).to_untyped(),
|
|
has_min: c.min.is_some(),
|
|
has_max: c.max.is_some(),
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_window_adapter_new(
|
|
user_data: WindowAdapterUserData,
|
|
drop: unsafe extern "C" fn(WindowAdapterUserData),
|
|
get_renderer_ref: unsafe extern "C" fn(WindowAdapterUserData) -> RendererPtr,
|
|
set_visible: unsafe extern "C" fn(WindowAdapterUserData, bool),
|
|
request_redraw: unsafe extern "C" fn(WindowAdapterUserData),
|
|
size: unsafe extern "C" fn(WindowAdapterUserData) -> IntSize,
|
|
set_size: unsafe extern "C" fn(WindowAdapterUserData, IntSize),
|
|
update_window_properties: unsafe extern "C" fn(WindowAdapterUserData, &WindowProperties),
|
|
position: unsafe extern "C" fn(
|
|
WindowAdapterUserData,
|
|
&mut euclid::default::Point2D<i32>,
|
|
) -> bool,
|
|
set_position: unsafe extern "C" fn(WindowAdapterUserData, euclid::default::Point2D<i32>),
|
|
target: *mut WindowAdapterRcOpaque,
|
|
) {
|
|
let window = Rc::<CppWindowAdapter>::new_cyclic(|w| CppWindowAdapter {
|
|
window: Window::new(w.clone()),
|
|
user_data,
|
|
drop,
|
|
get_renderer_ref,
|
|
set_visible,
|
|
request_redraw,
|
|
size,
|
|
set_size,
|
|
update_window_properties,
|
|
position,
|
|
set_position,
|
|
});
|
|
|
|
core::ptr::write(target as *mut Rc<dyn WindowAdapter>, window);
|
|
}
|
|
|
|
type PlatformUserData = *mut c_void;
|
|
|
|
struct CppPlatform {
|
|
user_data: PlatformUserData,
|
|
drop: unsafe extern "C" fn(PlatformUserData),
|
|
window_factory: unsafe extern "C" fn(PlatformUserData, *mut WindowAdapterRcOpaque),
|
|
#[cfg(not(feature = "std"))]
|
|
duration_since_start: unsafe extern "C" fn(PlatformUserData) -> u64,
|
|
// silent the warning despite `Clipboard` is a `#[non_exhaustive]` enum from another crate.
|
|
#[allow(improper_ctypes_definitions)]
|
|
set_clipboard_text: unsafe extern "C" fn(PlatformUserData, &SharedString, Clipboard),
|
|
#[allow(improper_ctypes_definitions)]
|
|
clipboard_text: unsafe extern "C" fn(PlatformUserData, &mut SharedString, Clipboard) -> bool,
|
|
run_event_loop: unsafe extern "C" fn(PlatformUserData),
|
|
quit_event_loop: unsafe extern "C" fn(PlatformUserData),
|
|
invoke_from_event_loop: unsafe extern "C" fn(PlatformUserData, PlatformTaskOpaque),
|
|
}
|
|
|
|
impl Drop for CppPlatform {
|
|
fn drop(&mut self) {
|
|
unsafe { (self.drop)(self.user_data) };
|
|
}
|
|
}
|
|
|
|
impl Platform for CppPlatform {
|
|
fn create_window_adapter(&self) -> Result<Rc<dyn WindowAdapter>, PlatformError> {
|
|
let mut uninit = core::mem::MaybeUninit::<Rc<dyn WindowAdapter>>::uninit();
|
|
unsafe {
|
|
(self.window_factory)(
|
|
self.user_data,
|
|
uninit.as_mut_ptr() as *mut WindowAdapterRcOpaque,
|
|
);
|
|
Ok(uninit.assume_init())
|
|
}
|
|
}
|
|
|
|
#[cfg(not(feature = "std"))]
|
|
fn duration_since_start(&self) -> core::time::Duration {
|
|
core::time::Duration::from_millis(unsafe { (self.duration_since_start)(self.user_data) })
|
|
}
|
|
|
|
fn run_event_loop(&self) -> Result<(), PlatformError> {
|
|
unsafe { (self.run_event_loop)(self.user_data) };
|
|
Ok(())
|
|
}
|
|
|
|
fn new_event_loop_proxy(&self) -> Option<Box<dyn i_slint_core::platform::EventLoopProxy>> {
|
|
Some(Box::new(CppEventLoopProxy {
|
|
user_data: self.user_data,
|
|
quit_event_loop: self.quit_event_loop,
|
|
invoke_from_event_loop: self.invoke_from_event_loop,
|
|
}))
|
|
}
|
|
|
|
fn set_clipboard_text(&self, text: &str, clipboard: Clipboard) {
|
|
let shared_text = SharedString::from(text);
|
|
unsafe { (self.set_clipboard_text)(self.user_data, &shared_text, clipboard) }
|
|
}
|
|
|
|
fn clipboard_text(&self, clipboard: Clipboard) -> Option<String> {
|
|
let mut out_text = SharedString::new();
|
|
let status = unsafe { (self.clipboard_text)(self.user_data, &mut out_text, clipboard) };
|
|
status.then(|| out_text.into())
|
|
}
|
|
|
|
#[cfg(feature = "esp-println")]
|
|
fn debug_log(&self, arguments: core::fmt::Arguments) {
|
|
esp_println::println!("{arguments}");
|
|
}
|
|
}
|
|
|
|
struct CppEventLoopProxy {
|
|
user_data: PlatformUserData,
|
|
quit_event_loop: unsafe extern "C" fn(PlatformUserData),
|
|
invoke_from_event_loop: unsafe extern "C" fn(PlatformUserData, PlatformTaskOpaque),
|
|
}
|
|
|
|
impl i_slint_core::platform::EventLoopProxy for CppEventLoopProxy {
|
|
fn quit_event_loop(&self) -> Result<(), i_slint_core::api::EventLoopError> {
|
|
unsafe { (self.quit_event_loop)(self.user_data) };
|
|
Ok(())
|
|
}
|
|
|
|
fn invoke_from_event_loop(
|
|
&self,
|
|
event: Box<dyn FnOnce() + Send>,
|
|
) -> Result<(), i_slint_core::api::EventLoopError> {
|
|
unsafe {
|
|
(self.invoke_from_event_loop)(
|
|
self.user_data,
|
|
core::mem::transmute::<*mut dyn FnOnce(), PlatformTaskOpaque>(Box::into_raw(event)),
|
|
)
|
|
};
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
unsafe impl Send for CppEventLoopProxy {}
|
|
unsafe impl Sync for CppEventLoopProxy {}
|
|
|
|
// silent the warning depite `Clipboard` is a `#[non_exhaustive]` enum from another crate.
|
|
#[allow(improper_ctypes_definitions)]
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_platform_register(
|
|
user_data: PlatformUserData,
|
|
drop: unsafe extern "C" fn(PlatformUserData),
|
|
window_factory: unsafe extern "C" fn(PlatformUserData, *mut WindowAdapterRcOpaque),
|
|
#[allow(unused)] duration_since_start: unsafe extern "C" fn(PlatformUserData) -> u64,
|
|
set_clipboard_text: unsafe extern "C" fn(PlatformUserData, &SharedString, Clipboard),
|
|
clipboard_text: unsafe extern "C" fn(PlatformUserData, &mut SharedString, Clipboard) -> bool,
|
|
run_event_loop: unsafe extern "C" fn(PlatformUserData),
|
|
quit_event_loop: unsafe extern "C" fn(PlatformUserData),
|
|
invoke_from_event_loop: unsafe extern "C" fn(PlatformUserData, PlatformTaskOpaque),
|
|
) {
|
|
let p = CppPlatform {
|
|
user_data,
|
|
drop,
|
|
window_factory,
|
|
#[cfg(not(feature = "std"))]
|
|
duration_since_start,
|
|
set_clipboard_text,
|
|
clipboard_text,
|
|
run_event_loop,
|
|
quit_event_loop,
|
|
invoke_from_event_loop,
|
|
};
|
|
i_slint_core::platform::set_platform(Box::new(p)).unwrap();
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_windowrc_has_active_animations(
|
|
handle: *const WindowAdapterRcOpaque,
|
|
) -> bool {
|
|
let window_adapter = &*(handle as *const Rc<dyn WindowAdapter>);
|
|
window_adapter.window().has_active_animations()
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_platform_update_timers_and_animations() {
|
|
i_slint_core::platform::update_timers_and_animations()
|
|
}
|
|
|
|
/// Returns the duration in millisecond until the next timer or `u64::MAX` if there is no pending timers
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_platform_duration_until_next_timer_update() -> u64 {
|
|
i_slint_core::platform::duration_until_next_timer_update()
|
|
.map_or(u64::MAX, |d| d.as_millis() as u64)
|
|
}
|
|
|
|
#[repr(C)]
|
|
pub struct PlatformTaskOpaque(*const c_void, *const c_void);
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_platform_task_drop(event: PlatformTaskOpaque) {
|
|
drop(Box::from_raw(core::mem::transmute::<PlatformTaskOpaque, *mut dyn FnOnce()>(event)));
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_platform_task_run(event: PlatformTaskOpaque) {
|
|
let f = Box::from_raw(core::mem::transmute::<PlatformTaskOpaque, *mut dyn FnOnce()>(event));
|
|
f();
|
|
}
|
|
|
|
#[cfg(feature = "renderer-software")]
|
|
mod software_renderer {
|
|
use super::*;
|
|
type SoftwareRendererOpaque = *const c_void;
|
|
use i_slint_core::graphics::{IntRect, Rgb8Pixel};
|
|
use i_slint_core::software_renderer::{
|
|
PhysicalRegion, RepaintBufferType, Rgb565Pixel, SoftwareRenderer,
|
|
};
|
|
use i_slint_core::SharedVector;
|
|
|
|
#[cfg(feature = "experimental")]
|
|
use i_slint_core::software_renderer::{TargetPixelBuffer, TexturePixelFormat};
|
|
|
|
#[cfg(feature = "experimental")]
|
|
type CppTargetPixelBufferUserData = *mut c_void;
|
|
|
|
#[cfg(feature = "experimental")]
|
|
#[repr(C)]
|
|
pub struct DrawTextureArgs {
|
|
pub image_data: *const u8,
|
|
pub pixel_format: TexturePixelFormat,
|
|
pub byte_stride: usize,
|
|
pub width: u32,
|
|
pub height: u32,
|
|
|
|
pub colorize: i_slint_core::Color,
|
|
pub alpha: u8,
|
|
|
|
pub dst_x: isize,
|
|
pub dst_y: isize,
|
|
pub dst_width: usize,
|
|
pub dst_height: usize,
|
|
/// 0, 90, 180, or 270
|
|
pub rotation: i32,
|
|
|
|
pub has_tiling: bool,
|
|
|
|
pub tiling_offset_x: i32,
|
|
pub tiling_offset_y: i32,
|
|
pub tiling_scale_x: f32,
|
|
pub tiling_scale_y: f32,
|
|
pub tiling_gap_x: u32,
|
|
pub tiling_gap_y: u32,
|
|
}
|
|
#[cfg(feature = "experimental")]
|
|
impl From<&i_slint_core::software_renderer::DrawTextureArgs> for DrawTextureArgs {
|
|
fn from(from: &i_slint_core::software_renderer::DrawTextureArgs) -> Self {
|
|
let source = from.source();
|
|
Self {
|
|
image_data: source.data.as_ptr(),
|
|
pixel_format: source.pixel_format,
|
|
byte_stride: source.byte_stride,
|
|
width: source.width,
|
|
height: source.height,
|
|
colorize: from.colorize.unwrap_or_default(),
|
|
alpha: from.alpha,
|
|
dst_x: from.dst_x,
|
|
dst_y: from.dst_y,
|
|
dst_width: from.dst_width,
|
|
dst_height: from.dst_height,
|
|
rotation: from.rotation.angle() as _,
|
|
has_tiling: from.tiling.is_some(),
|
|
tiling_offset_x: from.tiling.as_ref().map(|t| t.offset_x).unwrap_or_default(),
|
|
tiling_offset_y: from.tiling.as_ref().map(|t| t.offset_y).unwrap_or_default(),
|
|
tiling_scale_x: from.tiling.as_ref().map(|t| t.scale_x).unwrap_or_default(),
|
|
tiling_scale_y: from.tiling.as_ref().map(|t| t.scale_y).unwrap_or_default(),
|
|
tiling_gap_x: from.tiling.as_ref().map(|t| t.gap_x).unwrap_or_default(),
|
|
tiling_gap_y: from.tiling.as_ref().map(|t| t.gap_y).unwrap_or_default(),
|
|
}
|
|
}
|
|
}
|
|
|
|
#[cfg(feature = "experimental")]
|
|
#[repr(C)]
|
|
pub struct DrawRectangleArgs {
|
|
pub x: f32,
|
|
pub y: f32,
|
|
pub width: f32,
|
|
pub height: f32,
|
|
|
|
pub top_left_radius: f32,
|
|
pub top_right_radius: f32,
|
|
pub bottom_right_radius: f32,
|
|
pub bottom_left_radius: f32,
|
|
|
|
pub border_width: f32,
|
|
|
|
pub background: Brush,
|
|
pub border: Brush,
|
|
|
|
pub alpha: u8,
|
|
/// 0, 90, 180, or 270
|
|
pub rotation: i32,
|
|
}
|
|
#[cfg(feature = "experimental")]
|
|
impl From<&i_slint_core::software_renderer::DrawRectangleArgs> for DrawRectangleArgs {
|
|
fn from(from: &i_slint_core::software_renderer::DrawRectangleArgs) -> Self {
|
|
Self {
|
|
x: from.x,
|
|
y: from.y,
|
|
width: from.width,
|
|
height: from.height,
|
|
top_left_radius: from.top_left_radius,
|
|
top_right_radius: from.top_right_radius,
|
|
bottom_right_radius: from.bottom_right_radius,
|
|
bottom_left_radius: from.bottom_left_radius,
|
|
border_width: from.border_width,
|
|
background: from.background.clone(),
|
|
border: from.border.clone(),
|
|
alpha: from.alpha,
|
|
rotation: from.rotation.angle() as _,
|
|
}
|
|
}
|
|
}
|
|
|
|
#[repr(C)]
|
|
#[cfg(feature = "experimental")]
|
|
pub struct CppTargetPixelBuffer<T> {
|
|
user_data: CppTargetPixelBufferUserData,
|
|
line_slice: unsafe extern "C" fn(
|
|
CppTargetPixelBufferUserData,
|
|
usize,
|
|
slice_ptr: &mut *mut T,
|
|
slice_len: *mut usize,
|
|
),
|
|
num_lines: unsafe extern "C" fn(CppTargetPixelBufferUserData) -> usize,
|
|
fill_background:
|
|
unsafe extern "C" fn(CppTargetPixelBufferUserData, &Brush, &PhysicalRegion) -> bool,
|
|
draw_rectangle: unsafe extern "C" fn(
|
|
CppTargetPixelBufferUserData,
|
|
&DrawRectangleArgs,
|
|
&PhysicalRegion,
|
|
) -> bool,
|
|
draw_texture: unsafe extern "C" fn(
|
|
CppTargetPixelBufferUserData,
|
|
&DrawTextureArgs,
|
|
&PhysicalRegion,
|
|
) -> bool,
|
|
}
|
|
|
|
#[cfg(feature = "experimental")]
|
|
impl<TargetPixel: i_slint_core::software_renderer::TargetPixel> TargetPixelBuffer
|
|
for CppTargetPixelBuffer<TargetPixel>
|
|
{
|
|
type TargetPixel = TargetPixel;
|
|
|
|
fn line_slice(&mut self, line_number: usize) -> &mut [Self::TargetPixel] {
|
|
unsafe {
|
|
let mut data = core::ptr::null_mut();
|
|
let mut len = 0;
|
|
(self.line_slice)(self.user_data, line_number, &mut data, &mut len);
|
|
core::slice::from_raw_parts_mut(data, len)
|
|
}
|
|
}
|
|
|
|
fn num_lines(&self) -> usize {
|
|
unsafe { (self.num_lines)(self.user_data) }
|
|
}
|
|
|
|
/// Fill the background of the buffer with the given brush.
|
|
fn fill_background(
|
|
&mut self,
|
|
brush: &i_slint_core::Brush,
|
|
region: &PhysicalRegion,
|
|
) -> bool {
|
|
unsafe { (self.fill_background)(self.user_data, brush, region) }
|
|
}
|
|
|
|
/// Draw a rectangle specified by the DrawRectangleArgs. That rectangle must be clipped to the given region
|
|
fn draw_rectangle(
|
|
&mut self,
|
|
args: &i_slint_core::software_renderer::DrawRectangleArgs,
|
|
clip: &PhysicalRegion,
|
|
) -> bool {
|
|
let args = args.into();
|
|
unsafe { (self.draw_rectangle)(self.user_data, &args, clip) }
|
|
}
|
|
|
|
fn draw_texture(
|
|
&mut self,
|
|
texture: &i_slint_core::software_renderer::DrawTextureArgs,
|
|
clip: &PhysicalRegion,
|
|
) -> bool {
|
|
let texture = texture.into();
|
|
unsafe { (self.draw_texture)(self.user_data, &texture, clip) }
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_software_renderer_new(
|
|
buffer_age: u32,
|
|
) -> SoftwareRendererOpaque {
|
|
let repaint_buffer_type = match buffer_age {
|
|
0 => RepaintBufferType::NewBuffer,
|
|
1 => RepaintBufferType::ReusedBuffer,
|
|
2 => RepaintBufferType::SwappedBuffers,
|
|
_ => unreachable!(),
|
|
};
|
|
Box::into_raw(Box::new(SoftwareRenderer::new_with_repaint_buffer_type(repaint_buffer_type)))
|
|
as SoftwareRendererOpaque
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_software_renderer_drop(r: SoftwareRendererOpaque) {
|
|
drop(Box::from_raw(r as *mut SoftwareRenderer));
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_software_renderer_render_rgb8(
|
|
r: SoftwareRendererOpaque,
|
|
buffer: *mut Rgb8Pixel,
|
|
buffer_len: usize,
|
|
pixel_stride: usize,
|
|
) -> PhysicalRegion {
|
|
let buffer = core::slice::from_raw_parts_mut(buffer, buffer_len);
|
|
let renderer = &*(r as *const SoftwareRenderer);
|
|
renderer.render(buffer, pixel_stride)
|
|
}
|
|
|
|
#[cfg(feature = "experimental")]
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_software_renderer_render_accel_rgb8(
|
|
r: SoftwareRendererOpaque,
|
|
buffer: *mut CppTargetPixelBuffer<Rgb8Pixel>,
|
|
) -> PhysicalRegion {
|
|
let renderer = &*(r as *const SoftwareRenderer);
|
|
unsafe { renderer.render_into_buffer(&mut *buffer) }
|
|
}
|
|
|
|
#[cfg(feature = "experimental")]
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_software_renderer_render_accel_rgb565(
|
|
r: SoftwareRendererOpaque,
|
|
buffer: *mut CppTargetPixelBuffer<Rgb565Pixel>,
|
|
) -> PhysicalRegion {
|
|
let renderer = &*(r as *const SoftwareRenderer);
|
|
unsafe { renderer.render_into_buffer(&mut *buffer) }
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_software_renderer_render_rgb565(
|
|
r: SoftwareRendererOpaque,
|
|
buffer: *mut u16,
|
|
buffer_len: usize,
|
|
pixel_stride: usize,
|
|
) -> PhysicalRegion {
|
|
let buffer = core::slice::from_raw_parts_mut(buffer as *mut Rgb565Pixel, buffer_len);
|
|
let renderer = &*(r as *const SoftwareRenderer);
|
|
renderer.render(buffer, pixel_stride)
|
|
}
|
|
|
|
struct LineByLineProcessor<TargetPixel> {
|
|
process_line_fn: extern "C" fn(
|
|
*mut core::ffi::c_void,
|
|
usize,
|
|
usize,
|
|
usize,
|
|
extern "C" fn(*const core::ffi::c_void, *mut TargetPixel, usize),
|
|
*const core::ffi::c_void,
|
|
),
|
|
user_data: *mut core::ffi::c_void,
|
|
}
|
|
|
|
impl<TargetPixel: i_slint_core::software_renderer::TargetPixel>
|
|
i_slint_core::software_renderer::LineBufferProvider for LineByLineProcessor<TargetPixel>
|
|
{
|
|
type TargetPixel = TargetPixel;
|
|
fn process_line(
|
|
&mut self,
|
|
line: usize,
|
|
range: core::ops::Range<usize>,
|
|
render_fn: impl FnOnce(&mut [TargetPixel]),
|
|
) {
|
|
self.cpp_process_line(line, range, render_fn);
|
|
}
|
|
}
|
|
|
|
impl<TargetPixel> LineByLineProcessor<TargetPixel> {
|
|
fn cpp_process_line<RenderFn: FnOnce(&mut [TargetPixel])>(
|
|
&mut self,
|
|
line: usize,
|
|
range: core::ops::Range<usize>,
|
|
render_fn: RenderFn,
|
|
) {
|
|
let mut render_fn = Some(render_fn);
|
|
let render_fn_ptr = &mut render_fn as *mut Option<RenderFn> as *const core::ffi::c_void;
|
|
|
|
extern "C" fn cpp_render_line_callback<
|
|
TargetPixel,
|
|
RenderFn: FnOnce(&mut [TargetPixel]),
|
|
>(
|
|
render_fn_ptr: *const core::ffi::c_void,
|
|
line_start: *mut TargetPixel,
|
|
len: usize,
|
|
) {
|
|
let line_slice = unsafe { core::slice::from_raw_parts_mut(line_start, len) };
|
|
let render_fn =
|
|
unsafe { (*(render_fn_ptr as *mut Option<RenderFn>)).take().unwrap() };
|
|
render_fn(line_slice);
|
|
}
|
|
|
|
(self.process_line_fn)(
|
|
self.user_data,
|
|
line,
|
|
range.start,
|
|
range.end,
|
|
cpp_render_line_callback::<TargetPixel, RenderFn>,
|
|
render_fn_ptr,
|
|
);
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_software_renderer_render_by_line_rgb565(
|
|
r: SoftwareRendererOpaque,
|
|
process_line_fn: extern "C" fn(
|
|
*mut core::ffi::c_void,
|
|
usize,
|
|
usize,
|
|
usize,
|
|
extern "C" fn(*const core::ffi::c_void, *mut Rgb565Pixel, usize),
|
|
*const core::ffi::c_void,
|
|
),
|
|
user_data: *mut core::ffi::c_void,
|
|
) -> PhysicalRegion {
|
|
let renderer = &*(r as *const SoftwareRenderer);
|
|
let processor = LineByLineProcessor { process_line_fn, user_data };
|
|
renderer.render_by_line(processor)
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_software_renderer_render_by_line_rgb8(
|
|
r: SoftwareRendererOpaque,
|
|
process_line_fn: extern "C" fn(
|
|
*mut core::ffi::c_void,
|
|
usize,
|
|
usize,
|
|
usize,
|
|
extern "C" fn(*const core::ffi::c_void, *mut Rgb8Pixel, usize),
|
|
*const core::ffi::c_void,
|
|
),
|
|
user_data: *mut core::ffi::c_void,
|
|
) -> PhysicalRegion {
|
|
let renderer = &*(r as *const SoftwareRenderer);
|
|
let processor = LineByLineProcessor { process_line_fn, user_data };
|
|
renderer.render_by_line(processor)
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_software_renderer_set_rendering_rotation(
|
|
r: SoftwareRendererOpaque,
|
|
rotation: i32,
|
|
) {
|
|
use i_slint_core::software_renderer::RenderingRotation;
|
|
let renderer = &*(r as *const SoftwareRenderer);
|
|
renderer.set_rendering_rotation(match rotation {
|
|
90 => RenderingRotation::Rotate90,
|
|
180 => RenderingRotation::Rotate180,
|
|
270 => RenderingRotation::Rotate270,
|
|
_ => RenderingRotation::NoRotation,
|
|
});
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_software_renderer_handle(
|
|
r: SoftwareRendererOpaque,
|
|
) -> RendererPtr {
|
|
let r = (r as *const SoftwareRenderer) as *const dyn Renderer;
|
|
core::mem::transmute(r)
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_software_renderer_region_to_rects(
|
|
region: &PhysicalRegion,
|
|
out: &mut SharedVector<IntRect>,
|
|
) {
|
|
*out = region
|
|
.iter()
|
|
.map(|r| euclid::rect(r.0.x, r.0.y, r.1.width as i32, r.1.height as i32))
|
|
.collect();
|
|
}
|
|
}
|
|
|
|
#[cfg(all(feature = "i-slint-renderer-skia", feature = "raw-window-handle"))]
|
|
pub mod skia {
|
|
use super::*;
|
|
use raw_window_handle::{RawDisplayHandle, RawWindowHandle};
|
|
use std::sync::Arc;
|
|
|
|
struct RawHandlePair((RawWindowHandle, RawDisplayHandle));
|
|
|
|
impl raw_window_handle::HasDisplayHandle for RawHandlePair {
|
|
fn display_handle(
|
|
&self,
|
|
) -> Result<raw_window_handle::DisplayHandle<'_>, raw_window_handle::HandleError> {
|
|
// Safety: It is assumed that the C++ side keeps the window/display handles alive.
|
|
Ok(unsafe { raw_window_handle::DisplayHandle::borrow_raw(self.0 .1) })
|
|
}
|
|
}
|
|
|
|
impl raw_window_handle::HasWindowHandle for RawHandlePair {
|
|
fn window_handle(
|
|
&self,
|
|
) -> Result<raw_window_handle::WindowHandle<'_>, raw_window_handle::HandleError> {
|
|
// Safety: It is assumed that the C++ side keeps the window/display handles alive.
|
|
Ok(unsafe { raw_window_handle::WindowHandle::borrow_raw(self.0 .0) })
|
|
}
|
|
}
|
|
|
|
struct CppRawHandle(Arc<RawHandlePair>);
|
|
|
|
impl From<(RawWindowHandle, RawDisplayHandle)> for CppRawHandle {
|
|
fn from(pair: (RawWindowHandle, RawDisplayHandle)) -> Self {
|
|
Self(Arc::new(RawHandlePair(pair)))
|
|
}
|
|
}
|
|
|
|
type CppRawHandleOpaque = *const c_void;
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_new_raw_window_handle_win32(
|
|
hwnd: *mut c_void,
|
|
_hinstance: *mut c_void,
|
|
) -> CppRawHandleOpaque {
|
|
let handle = CppRawHandle::from((
|
|
RawWindowHandle::Win32(raw_window_handle::Win32WindowHandle::new(
|
|
(hwnd as isize).try_into().expect("C++: NativeWindowHandle created with null hwnd"),
|
|
)),
|
|
RawDisplayHandle::Windows(raw_window_handle::WindowsDisplayHandle::new()),
|
|
));
|
|
Box::into_raw(Box::new(handle)) as CppRawHandleOpaque
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_new_raw_window_handle_x11_xcb(
|
|
window: u32,
|
|
visual_id: u32,
|
|
connection: *mut c_void,
|
|
screen: core::ffi::c_int,
|
|
) -> CppRawHandleOpaque {
|
|
use raw_window_handle::{XcbDisplayHandle, XcbWindowHandle};
|
|
let handle = CppRawHandle::from((
|
|
RawWindowHandle::Xcb({
|
|
let mut hnd = XcbWindowHandle::new(
|
|
window
|
|
.try_into()
|
|
.expect("C++: NativeWindowHandle created with null xcb window handle"),
|
|
);
|
|
hnd.visual_id = visual_id.try_into().ok();
|
|
hnd
|
|
}),
|
|
RawDisplayHandle::Xcb(XcbDisplayHandle::new(
|
|
Some(core::ptr::NonNull::new_unchecked(connection)),
|
|
screen,
|
|
)),
|
|
));
|
|
Box::into_raw(Box::new(handle)) as CppRawHandleOpaque
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_new_raw_window_handle_x11_xlib(
|
|
window: core::ffi::c_ulong,
|
|
visual_id: core::ffi::c_ulong,
|
|
display: *mut c_void,
|
|
screen: core::ffi::c_int,
|
|
) -> CppRawHandleOpaque {
|
|
use raw_window_handle::{XlibDisplayHandle, XlibWindowHandle};
|
|
let handle = CppRawHandle::from((
|
|
RawWindowHandle::Xlib({
|
|
let mut hnd = XlibWindowHandle::new(window);
|
|
hnd.visual_id = visual_id;
|
|
hnd
|
|
}),
|
|
RawDisplayHandle::Xlib(XlibDisplayHandle::new(
|
|
Some(core::ptr::NonNull::new_unchecked(display)),
|
|
screen,
|
|
)),
|
|
));
|
|
Box::into_raw(Box::new(handle)) as CppRawHandleOpaque
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_new_raw_window_handle_wayland(
|
|
surface: *mut c_void,
|
|
display: *mut c_void,
|
|
) -> CppRawHandleOpaque {
|
|
use raw_window_handle::{WaylandDisplayHandle, WaylandWindowHandle};
|
|
let handle = CppRawHandle::from((
|
|
RawWindowHandle::Wayland(WaylandWindowHandle::new(core::ptr::NonNull::new_unchecked(
|
|
surface,
|
|
))),
|
|
RawDisplayHandle::Wayland(WaylandDisplayHandle::new(
|
|
core::ptr::NonNull::new_unchecked(display),
|
|
)),
|
|
));
|
|
Box::into_raw(Box::new(handle)) as CppRawHandleOpaque
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_new_raw_window_handle_appkit(
|
|
ns_view: *mut c_void,
|
|
_ns_window: *mut c_void,
|
|
) -> CppRawHandleOpaque {
|
|
use raw_window_handle::{AppKitDisplayHandle, AppKitWindowHandle};
|
|
let handle = CppRawHandle::from((
|
|
RawWindowHandle::AppKit(AppKitWindowHandle::new(core::ptr::NonNull::new_unchecked(
|
|
ns_view,
|
|
))),
|
|
RawDisplayHandle::AppKit(AppKitDisplayHandle::new()),
|
|
));
|
|
Box::into_raw(Box::new(handle)) as CppRawHandleOpaque
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_raw_window_handle_drop(handle: CppRawHandleOpaque) {
|
|
drop(Box::from_raw(handle as *mut CppRawHandle))
|
|
}
|
|
|
|
type SkiaRendererOpaque = *const c_void;
|
|
type SkiaRenderer = i_slint_renderer_skia::SkiaRenderer;
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_skia_renderer_new(
|
|
handle_opaque: CppRawHandleOpaque,
|
|
size: IntSize,
|
|
) -> SkiaRendererOpaque {
|
|
let handle = &*(handle_opaque as *const CppRawHandle);
|
|
|
|
let boxed_renderer: Box<SkiaRenderer> = Box::new(
|
|
SkiaRenderer::new(
|
|
&i_slint_renderer_skia::SkiaSharedContext::default(),
|
|
handle.0.clone(),
|
|
handle.0.clone(),
|
|
PhysicalSize { width: size.width, height: size.height },
|
|
)
|
|
.unwrap(),
|
|
);
|
|
Box::into_raw(boxed_renderer) as SkiaRendererOpaque
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_skia_renderer_drop(r: SkiaRendererOpaque) {
|
|
drop(Box::from_raw(r as *mut SkiaRenderer))
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_skia_renderer_render(r: SkiaRendererOpaque) {
|
|
let r = &*(r as *const SkiaRenderer);
|
|
r.render().unwrap();
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_skia_renderer_handle(r: SkiaRendererOpaque) -> RendererPtr {
|
|
let r = (r as *const SkiaRenderer) as *const dyn Renderer;
|
|
core::mem::transmute(r)
|
|
}
|
|
}
|