mirror of
				https://github.com/slint-ui/slint.git
				synced 2025-10-31 12:04:33 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			232 lines
		
	
	
	
		
			7 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			232 lines
		
	
	
	
		
			7 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
 | |
| 
 | |
| /*! This crate just expose the function used by the C++ integration */
 | |
| 
 | |
| #![no_std]
 | |
| extern crate alloc;
 | |
| #[cfg(feature = "std")]
 | |
| extern crate std;
 | |
| 
 | |
| use alloc::rc::Rc;
 | |
| use core::ffi::c_void;
 | |
| use i_slint_core::items::OperatingSystemType;
 | |
| use i_slint_core::window::{ffi::WindowAdapterRcOpaque, WindowAdapter};
 | |
| use i_slint_core::SharedString;
 | |
| 
 | |
| pub mod platform;
 | |
| 
 | |
| #[cfg(feature = "i-slint-backend-selector")]
 | |
| use i_slint_backend_selector::with_platform;
 | |
| 
 | |
| #[cfg(not(feature = "i-slint-backend-selector"))]
 | |
| pub fn with_platform<R>(
 | |
|     f: impl FnOnce(
 | |
|         &dyn i_slint_core::platform::Platform,
 | |
|     ) -> Result<R, i_slint_core::platform::PlatformError>,
 | |
| ) -> Result<R, i_slint_core::platform::PlatformError> {
 | |
|     i_slint_core::with_platform(|| Err(i_slint_core::platform::PlatformError::NoPlatform), f)
 | |
| }
 | |
| 
 | |
| // One need to make sure something from the crate is exported,
 | |
| // otherwise its symbols are not going to be in the final binary
 | |
| #[cfg(feature = "testing")]
 | |
| pub use i_slint_backend_testing;
 | |
| #[cfg(feature = "slint-interpreter")]
 | |
| pub use slint_interpreter;
 | |
| 
 | |
| #[unsafe(no_mangle)]
 | |
| pub unsafe extern "C" fn slint_windowrc_init(out: *mut WindowAdapterRcOpaque) {
 | |
|     assert_eq!(
 | |
|         core::mem::size_of::<Rc<dyn WindowAdapter>>(),
 | |
|         core::mem::size_of::<WindowAdapterRcOpaque>()
 | |
|     );
 | |
|     let win = with_platform(|b| b.create_window_adapter()).unwrap();
 | |
|     core::ptr::write(out as *mut Rc<dyn WindowAdapter>, win);
 | |
| }
 | |
| 
 | |
| #[unsafe(no_mangle)]
 | |
| pub unsafe extern "C" fn slint_ensure_backend() {
 | |
|     with_platform(|_b| {
 | |
|         // Nothing to do, just make sure a backend was created
 | |
|         Ok(())
 | |
|     })
 | |
|     .unwrap()
 | |
| }
 | |
| 
 | |
| #[unsafe(no_mangle)]
 | |
| /// Enters the main event loop.
 | |
| pub extern "C" fn slint_run_event_loop(quit_on_last_window_closed: bool) {
 | |
|     with_platform(|b| {
 | |
|         if !quit_on_last_window_closed {
 | |
|             #[allow(deprecated)]
 | |
|             b.set_event_loop_quit_on_last_window_closed(false);
 | |
|         }
 | |
|         b.run_event_loop()
 | |
|     })
 | |
|     .unwrap();
 | |
| }
 | |
| 
 | |
| /// Will execute the given functor in the main thread
 | |
| #[unsafe(no_mangle)]
 | |
| pub unsafe extern "C" fn slint_post_event(
 | |
|     event: extern "C" fn(user_data: *mut c_void),
 | |
|     user_data: *mut c_void,
 | |
|     drop_user_data: Option<extern "C" fn(*mut c_void)>,
 | |
| ) {
 | |
|     struct UserData {
 | |
|         user_data: *mut c_void,
 | |
|         drop_user_data: Option<extern "C" fn(*mut c_void)>,
 | |
|     }
 | |
|     impl Drop for UserData {
 | |
|         fn drop(&mut self) {
 | |
|             if let Some(x) = self.drop_user_data {
 | |
|                 x(self.user_data)
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     unsafe impl Send for UserData {}
 | |
|     let ud = UserData { user_data, drop_user_data };
 | |
| 
 | |
|     i_slint_core::api::invoke_from_event_loop(move || {
 | |
|         let ud = &ud;
 | |
|         event(ud.user_data);
 | |
|     })
 | |
|     .unwrap();
 | |
| }
 | |
| 
 | |
| #[unsafe(no_mangle)]
 | |
| pub unsafe extern "C" fn slint_quit_event_loop() {
 | |
|     i_slint_core::api::quit_event_loop().unwrap();
 | |
| }
 | |
| 
 | |
| #[cfg(feature = "std")]
 | |
| #[unsafe(no_mangle)]
 | |
| pub unsafe extern "C" fn slint_register_font_from_path(
 | |
|     win: *const WindowAdapterRcOpaque,
 | |
|     path: &SharedString,
 | |
|     error_str: &mut SharedString,
 | |
| ) {
 | |
|     let window_adapter = &*(win as *const Rc<dyn WindowAdapter>);
 | |
|     *error_str = match window_adapter
 | |
|         .renderer()
 | |
|         .register_font_from_path(std::path::Path::new(path.as_str()))
 | |
|     {
 | |
|         Ok(()) => Default::default(),
 | |
|         Err(err) => i_slint_core::string::ToSharedString::to_shared_string(&err),
 | |
|     };
 | |
| }
 | |
| 
 | |
| #[cfg(feature = "std")]
 | |
| #[unsafe(no_mangle)]
 | |
| pub unsafe extern "C" fn slint_register_font_from_data(
 | |
|     win: *const WindowAdapterRcOpaque,
 | |
|     data: i_slint_core::slice::Slice<'static, u8>,
 | |
|     error_str: &mut SharedString,
 | |
| ) {
 | |
|     let window_adapter = &*(win as *const Rc<dyn WindowAdapter>);
 | |
|     *error_str = match window_adapter.renderer().register_font_from_memory(data.as_slice()) {
 | |
|         Ok(()) => Default::default(),
 | |
|         Err(err) => i_slint_core::string::ToSharedString::to_shared_string(&err),
 | |
|     };
 | |
| }
 | |
| 
 | |
| #[unsafe(no_mangle)]
 | |
| pub unsafe extern "C" fn slint_register_bitmap_font(
 | |
|     win: *const WindowAdapterRcOpaque,
 | |
|     font_data: &'static i_slint_core::graphics::BitmapFont,
 | |
| ) {
 | |
|     let window_adapter = &*(win as *const Rc<dyn WindowAdapter>);
 | |
|     window_adapter.renderer().register_bitmap_font(font_data);
 | |
| }
 | |
| 
 | |
| #[unsafe(no_mangle)]
 | |
| pub extern "C" fn slint_string_to_float(string: &SharedString, value: &mut f32) -> bool {
 | |
|     match string.as_str().parse::<f32>() {
 | |
|         Ok(v) => {
 | |
|             *value = v;
 | |
|             true
 | |
|         }
 | |
|         Err(_) => false,
 | |
|     }
 | |
| }
 | |
| 
 | |
| #[unsafe(no_mangle)]
 | |
| pub extern "C" fn slint_string_character_count(string: &SharedString) -> usize {
 | |
|     unicode_segmentation::UnicodeSegmentation::graphemes(string.as_str(), true).count()
 | |
| }
 | |
| 
 | |
| #[unsafe(no_mangle)]
 | |
| pub extern "C" fn slint_string_to_usize(string: &SharedString, value: &mut usize) -> bool {
 | |
|     match string.as_str().parse::<usize>() {
 | |
|         Ok(v) => {
 | |
|             *value = v;
 | |
|             true
 | |
|         }
 | |
|         Err(_) => false,
 | |
|     }
 | |
| }
 | |
| 
 | |
| #[unsafe(no_mangle)]
 | |
| pub extern "C" fn slint_debug(string: &SharedString) {
 | |
|     i_slint_core::debug_log!("{string}");
 | |
| }
 | |
| 
 | |
| #[cfg(not(feature = "std"))]
 | |
| mod allocator {
 | |
|     use core::alloc::Layout;
 | |
|     use core::ffi::c_void;
 | |
|     extern "C" {
 | |
|         pub fn free(p: *mut c_void);
 | |
|         pub fn malloc(size: usize) -> *mut c_void;
 | |
|     }
 | |
| 
 | |
|     struct CAlloc;
 | |
|     unsafe impl core::alloc::GlobalAlloc for CAlloc {
 | |
|         unsafe fn alloc(&self, layout: Layout) -> *mut u8 {
 | |
|             let align = layout.align();
 | |
|             if align <= core::mem::size_of::<usize>() {
 | |
|                 malloc(layout.size()) as *mut u8
 | |
|             } else {
 | |
|                 // Ideally we'd use aligned_alloc, but that function caused heap corruption with esp-idf
 | |
|                 let ptr = malloc(layout.size() + align) as *mut u8;
 | |
|                 let shift = align - (ptr as usize % align);
 | |
|                 let ptr = ptr.add(shift);
 | |
|                 core::ptr::write(ptr.sub(1), shift as u8);
 | |
|                 ptr
 | |
|             }
 | |
|         }
 | |
|         unsafe fn dealloc(&self, ptr: *mut u8, layout: Layout) {
 | |
|             let align = layout.align();
 | |
|             if align <= core::mem::size_of::<usize>() {
 | |
|                 free(ptr as *mut c_void);
 | |
|             } else {
 | |
|                 let shift = core::ptr::read(ptr.sub(1)) as usize;
 | |
|                 free(ptr.sub(shift) as *mut c_void);
 | |
|             }
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     #[global_allocator]
 | |
|     static ALLOCATOR: CAlloc = CAlloc;
 | |
| }
 | |
| 
 | |
| #[cfg(all(not(feature = "std"), not(feature = "esp-backtrace")))]
 | |
| #[panic_handler]
 | |
| fn panic(_info: &core::panic::PanicInfo) -> ! {
 | |
|     loop {}
 | |
| }
 | |
| #[cfg(feature = "esp-backtrace")]
 | |
| use esp_backtrace as _;
 | |
| 
 | |
| #[unsafe(no_mangle)]
 | |
| pub unsafe extern "C" fn slint_set_xdg_app_id(_app_id: &SharedString) {
 | |
|     #[cfg(feature = "i-slint-backend-selector")]
 | |
|     i_slint_backend_selector::with_global_context(|ctx| ctx.set_xdg_app_id(_app_id.clone()))
 | |
|         .unwrap();
 | |
| }
 | |
| 
 | |
| #[unsafe(no_mangle)]
 | |
| pub unsafe extern "C" fn slint_detect_operating_system() -> OperatingSystemType {
 | |
|     i_slint_core::detect_operating_system()
 | |
| }
 | 
