mirror of
https://github.com/slint-ui/slint.git
synced 2025-08-26 21:34:08 +00:00
962 lines
32 KiB
Rust
962 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 crate::dynamic_item_tree::ErasedItemTreeBox;
|
|
|
|
use super::*;
|
|
use core::ptr::NonNull;
|
|
use i_slint_core::model::{Model, ModelNotify, SharedVectorModel};
|
|
use i_slint_core::slice::Slice;
|
|
use i_slint_core::window::WindowAdapter;
|
|
use std::ffi::c_void;
|
|
use vtable::VRef;
|
|
|
|
/// Construct a new Value in the given memory location
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_value_new() -> Box<Value> {
|
|
Box::new(Value::default())
|
|
}
|
|
|
|
/// Construct a new Value in the given memory location
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_value_clone(other: &Value) -> Box<Value> {
|
|
Box::new(other.clone())
|
|
}
|
|
|
|
/// Destruct the value in that memory location
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_value_destructor(val: Box<Value>) {
|
|
drop(val);
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_value_eq(a: &Value, b: &Value) -> bool {
|
|
a == b
|
|
}
|
|
|
|
/// Construct a new Value in the given memory location as string
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_value_new_string(str: &SharedString) -> Box<Value> {
|
|
Box::new(Value::String(str.clone()))
|
|
}
|
|
|
|
/// Construct a new Value in the given memory location as double
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_value_new_double(double: f64) -> Box<Value> {
|
|
Box::new(Value::Number(double))
|
|
}
|
|
|
|
/// Construct a new Value in the given memory location as bool
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_value_new_bool(b: bool) -> Box<Value> {
|
|
Box::new(Value::Bool(b))
|
|
}
|
|
|
|
/// Construct a new Value in the given memory location as array model
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_value_new_array_model(
|
|
a: &SharedVector<Box<Value>>,
|
|
) -> Box<Value> {
|
|
let vec = a.iter().map(|vb| vb.as_ref().clone()).collect::<SharedVector<_>>();
|
|
Box::new(Value::Model(ModelRc::new(SharedVectorModel::from(vec))))
|
|
}
|
|
|
|
/// Construct a new Value in the given memory location as Brush
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_value_new_brush(brush: &Brush) -> Box<Value> {
|
|
Box::new(Value::Brush(brush.clone()))
|
|
}
|
|
|
|
/// Construct a new Value in the given memory location as Struct
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_value_new_struct(struc: &StructOpaque) -> Box<Value> {
|
|
Box::new(Value::Struct(struc.as_struct().clone()))
|
|
}
|
|
|
|
/// Construct a new Value in the given memory location as image
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_value_new_image(img: &Image) -> Box<Value> {
|
|
Box::new(Value::Image(img.clone()))
|
|
}
|
|
|
|
/// Construct a new Value containing a model in the given memory location
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_value_new_model(
|
|
model: NonNull<u8>,
|
|
vtable: &ModelAdaptorVTable,
|
|
) -> Box<Value> {
|
|
Box::new(Value::Model(ModelRc::new(ModelAdaptorWrapper(vtable::VBox::from_raw(
|
|
NonNull::from(vtable),
|
|
model,
|
|
)))))
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_value_type(val: &Value) -> ValueType {
|
|
val.value_type()
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_value_to_string(val: &Value) -> Option<&SharedString> {
|
|
match val {
|
|
Value::String(v) => Some(v),
|
|
_ => None,
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_value_to_number(val: &Value) -> Option<&f64> {
|
|
match val {
|
|
Value::Number(v) => Some(v),
|
|
_ => None,
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_value_to_bool(val: &Value) -> Option<&bool> {
|
|
match val {
|
|
Value::Bool(v) => Some(v),
|
|
_ => None,
|
|
}
|
|
}
|
|
|
|
/// Extracts a `SharedVector<ValueOpaque>` out of the given value `val`, writes that into the
|
|
/// `out` parameter and returns true; returns false if the value does not hold an extractable
|
|
/// array.
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_value_to_array(
|
|
val: &Box<Value>,
|
|
out: *mut SharedVector<Box<Value>>,
|
|
) -> bool {
|
|
match val.as_ref() {
|
|
Value::Model(m) => {
|
|
let vec = m.iter().map(|vb| Box::new(vb)).collect::<SharedVector<_>>();
|
|
unsafe {
|
|
std::ptr::write(out, vec);
|
|
}
|
|
|
|
true
|
|
}
|
|
_ => false,
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_value_to_brush(val: &Value) -> Option<&Brush> {
|
|
match val {
|
|
Value::Brush(b) => Some(b),
|
|
_ => None,
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_value_to_struct(val: &Value) -> *const StructOpaque {
|
|
match val {
|
|
Value::Struct(s) => s as *const Struct as *const StructOpaque,
|
|
_ => std::ptr::null(),
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_value_to_image(val: &Value) -> Option<&Image> {
|
|
match val {
|
|
Value::Image(img) => Some(img),
|
|
_ => None,
|
|
}
|
|
}
|
|
|
|
#[repr(C)]
|
|
#[cfg(target_pointer_width = "64")]
|
|
pub struct StructOpaque([usize; 6]);
|
|
#[repr(C)]
|
|
#[cfg(target_pointer_width = "32")]
|
|
pub struct StructOpaque([u64; 4]);
|
|
const _: [(); std::mem::size_of::<StructOpaque>()] = [(); std::mem::size_of::<Struct>()];
|
|
const _: [(); std::mem::align_of::<StructOpaque>()] = [(); std::mem::align_of::<Struct>()];
|
|
|
|
impl StructOpaque {
|
|
fn as_struct(&self) -> &Struct {
|
|
// Safety: there should be no way to construct a StructOpaque without it holding an actual Struct
|
|
unsafe { std::mem::transmute::<&StructOpaque, &Struct>(self) }
|
|
}
|
|
fn as_struct_mut(&mut self) -> &mut Struct {
|
|
// Safety: there should be no way to construct a StructOpaque without it holding an actual Struct
|
|
unsafe { std::mem::transmute::<&mut StructOpaque, &mut Struct>(self) }
|
|
}
|
|
}
|
|
|
|
/// Construct a new Struct in the given memory location
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_struct_new(val: *mut StructOpaque) {
|
|
std::ptr::write(val as *mut Struct, Struct::default())
|
|
}
|
|
|
|
/// Construct a new Struct in the given memory location
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_struct_clone(
|
|
other: &StructOpaque,
|
|
val: *mut StructOpaque,
|
|
) {
|
|
std::ptr::write(val as *mut Struct, other.as_struct().clone())
|
|
}
|
|
|
|
/// Destruct the struct in that memory location
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_struct_destructor(val: *mut StructOpaque) {
|
|
drop(std::ptr::read(val as *mut Struct))
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_struct_get_field(
|
|
stru: &StructOpaque,
|
|
name: Slice<u8>,
|
|
) -> *mut Value {
|
|
if let Some(value) = stru.as_struct().get_field(std::str::from_utf8(&name).unwrap()) {
|
|
Box::into_raw(Box::new(value.clone()))
|
|
} else {
|
|
std::ptr::null_mut()
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_struct_set_field<'a>(
|
|
stru: &'a mut StructOpaque,
|
|
name: Slice<u8>,
|
|
value: &Value,
|
|
) {
|
|
stru.as_struct_mut().set_field(std::str::from_utf8(&name).unwrap().into(), value.clone())
|
|
}
|
|
|
|
type StructIterator<'a> = std::collections::hash_map::Iter<'a, SmolStr, Value>;
|
|
#[repr(C)]
|
|
pub struct StructIteratorOpaque<'a>([usize; 5], std::marker::PhantomData<StructIterator<'a>>);
|
|
const _: [(); std::mem::size_of::<StructIteratorOpaque>()] =
|
|
[(); std::mem::size_of::<StructIterator>()];
|
|
const _: [(); std::mem::align_of::<StructIteratorOpaque>()] =
|
|
[(); std::mem::align_of::<StructIterator>()];
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_struct_iterator_destructor(
|
|
val: *mut StructIteratorOpaque,
|
|
) {
|
|
drop(std::ptr::read(val as *mut StructIterator))
|
|
}
|
|
|
|
/// Advance the iterator and return the next value, or a null pointer
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_struct_iterator_next<'a>(
|
|
iter: &'a mut StructIteratorOpaque,
|
|
k: &mut Slice<'a, u8>,
|
|
) -> *mut Value {
|
|
if let Some((str, val)) = (*(iter as *mut StructIteratorOpaque as *mut StructIterator)).next() {
|
|
*k = Slice::from_slice(str.as_bytes());
|
|
Box::into_raw(Box::new(val.clone()))
|
|
} else {
|
|
*k = Slice::default();
|
|
std::ptr::null_mut()
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_struct_make_iter(
|
|
stru: &StructOpaque,
|
|
) -> StructIteratorOpaque<'_> {
|
|
let ret_it: StructIterator = stru.as_struct().0.iter();
|
|
unsafe {
|
|
let mut r = std::mem::MaybeUninit::<StructIteratorOpaque>::uninit();
|
|
std::ptr::write(r.as_mut_ptr() as *mut StructIterator, ret_it);
|
|
r.assume_init()
|
|
}
|
|
}
|
|
|
|
/// Get a property. Returns a null pointer if the property does not exist.
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_instance_get_property(
|
|
inst: &ErasedItemTreeBox,
|
|
name: Slice<u8>,
|
|
) -> *mut Value {
|
|
generativity::make_guard!(guard);
|
|
let comp = inst.unerase(guard);
|
|
match comp
|
|
.description()
|
|
.get_property(comp.borrow(), &normalize_identifier(std::str::from_utf8(&name).unwrap()))
|
|
{
|
|
Ok(val) => Box::into_raw(Box::new(val)),
|
|
Err(_) => std::ptr::null_mut(),
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_component_instance_set_property(
|
|
inst: &ErasedItemTreeBox,
|
|
name: Slice<u8>,
|
|
val: &Value,
|
|
) -> bool {
|
|
generativity::make_guard!(guard);
|
|
let comp = inst.unerase(guard);
|
|
comp.description()
|
|
.set_property(
|
|
comp.borrow(),
|
|
&normalize_identifier(std::str::from_utf8(&name).unwrap()),
|
|
val.clone(),
|
|
)
|
|
.is_ok()
|
|
}
|
|
|
|
/// Invoke a callback or function. Returns raw boxed value on success and null ptr on failure.
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_instance_invoke(
|
|
inst: &ErasedItemTreeBox,
|
|
name: Slice<u8>,
|
|
args: Slice<Box<Value>>,
|
|
) -> *mut Value {
|
|
let args = args.iter().map(|vb| vb.as_ref().clone()).collect::<Vec<_>>();
|
|
generativity::make_guard!(guard);
|
|
let comp = inst.unerase(guard);
|
|
match comp.description().invoke(
|
|
comp.borrow(),
|
|
&normalize_identifier(std::str::from_utf8(&name).unwrap()),
|
|
args.as_slice(),
|
|
) {
|
|
Ok(val) => Box::into_raw(Box::new(val)),
|
|
Err(_) => std::ptr::null_mut(),
|
|
}
|
|
}
|
|
|
|
/// Wrap the user_data provided by the native code and call the drop function on Drop.
|
|
///
|
|
/// Safety: user_data must be a pointer that can be destroyed by the drop_user_data function.
|
|
/// callback must be a valid callback that initialize the `ret`
|
|
struct CallbackUserData {
|
|
user_data: *mut c_void,
|
|
drop_user_data: Option<extern "C" fn(*mut c_void)>,
|
|
callback: extern "C" fn(user_data: *mut c_void, arg: Slice<Box<Value>>) -> Box<Value>,
|
|
}
|
|
|
|
impl Drop for CallbackUserData {
|
|
fn drop(&mut self) {
|
|
if let Some(x) = self.drop_user_data {
|
|
x(self.user_data)
|
|
}
|
|
}
|
|
}
|
|
|
|
impl CallbackUserData {
|
|
fn call(&self, args: &[Value]) -> Value {
|
|
let args = args.iter().map(|v| v.clone().into()).collect::<Vec<_>>();
|
|
(self.callback)(self.user_data, Slice::from_slice(args.as_ref())).as_ref().clone()
|
|
}
|
|
}
|
|
|
|
/// Set a handler for the callback.
|
|
/// The `callback` function must initialize the `ret` (the `ret` passed to the callback is initialized and is assumed initialized after the function)
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_instance_set_callback(
|
|
inst: &ErasedItemTreeBox,
|
|
name: Slice<u8>,
|
|
callback: extern "C" fn(user_data: *mut c_void, arg: Slice<Box<Value>>) -> Box<Value>,
|
|
user_data: *mut c_void,
|
|
drop_user_data: Option<extern "C" fn(*mut c_void)>,
|
|
) -> bool {
|
|
let ud = CallbackUserData { user_data, drop_user_data, callback };
|
|
|
|
generativity::make_guard!(guard);
|
|
let comp = inst.unerase(guard);
|
|
comp.description()
|
|
.set_callback_handler(
|
|
comp.borrow(),
|
|
&normalize_identifier(std::str::from_utf8(&name).unwrap()),
|
|
Box::new(move |args| ud.call(args)),
|
|
)
|
|
.is_ok()
|
|
}
|
|
|
|
/// Get a global property. Returns a raw boxed value on success; nullptr otherwise.
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_instance_get_global_property(
|
|
inst: &ErasedItemTreeBox,
|
|
global: Slice<u8>,
|
|
property_name: Slice<u8>,
|
|
) -> *mut Value {
|
|
generativity::make_guard!(guard);
|
|
let comp = inst.unerase(guard);
|
|
match comp
|
|
.description()
|
|
.get_global(comp.borrow(), &normalize_identifier(std::str::from_utf8(&global).unwrap()))
|
|
.and_then(|g| {
|
|
g.as_ref()
|
|
.get_property(&normalize_identifier(std::str::from_utf8(&property_name).unwrap()))
|
|
}) {
|
|
Ok(val) => Box::into_raw(Box::new(val)),
|
|
Err(_) => std::ptr::null_mut(),
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_component_instance_set_global_property(
|
|
inst: &ErasedItemTreeBox,
|
|
global: Slice<u8>,
|
|
property_name: Slice<u8>,
|
|
val: &Value,
|
|
) -> bool {
|
|
generativity::make_guard!(guard);
|
|
let comp = inst.unerase(guard);
|
|
comp.description()
|
|
.get_global(comp.borrow(), &normalize_identifier(std::str::from_utf8(&global).unwrap()))
|
|
.and_then(|g| {
|
|
g.as_ref()
|
|
.set_property(
|
|
&normalize_identifier(std::str::from_utf8(&property_name).unwrap()),
|
|
val.clone(),
|
|
)
|
|
.map_err(|_| ())
|
|
})
|
|
.is_ok()
|
|
}
|
|
|
|
/// The `callback` function must initialize the `ret` (the `ret` passed to the callback is initialized and is assumed initialized after the function)
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_instance_set_global_callback(
|
|
inst: &ErasedItemTreeBox,
|
|
global: Slice<u8>,
|
|
name: Slice<u8>,
|
|
callback: extern "C" fn(user_data: *mut c_void, arg: Slice<Box<Value>>) -> Box<Value>,
|
|
user_data: *mut c_void,
|
|
drop_user_data: Option<extern "C" fn(*mut c_void)>,
|
|
) -> bool {
|
|
let ud = CallbackUserData { user_data, drop_user_data, callback };
|
|
|
|
generativity::make_guard!(guard);
|
|
let comp = inst.unerase(guard);
|
|
comp.description()
|
|
.get_global(comp.borrow(), &normalize_identifier(std::str::from_utf8(&global).unwrap()))
|
|
.and_then(|g| {
|
|
g.as_ref().set_callback_handler(
|
|
&normalize_identifier(std::str::from_utf8(&name).unwrap()),
|
|
Box::new(move |args| ud.call(args)),
|
|
)
|
|
})
|
|
.is_ok()
|
|
}
|
|
|
|
/// Invoke a global callback or function. Returns raw boxed value on success; nullptr otherwise.
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_instance_invoke_global(
|
|
inst: &ErasedItemTreeBox,
|
|
global: Slice<u8>,
|
|
callable_name: Slice<u8>,
|
|
args: Slice<Box<Value>>,
|
|
) -> *mut Value {
|
|
let args = args.iter().map(|vb| vb.as_ref().clone()).collect::<Vec<_>>();
|
|
generativity::make_guard!(guard);
|
|
let comp = inst.unerase(guard);
|
|
let callable_name = std::str::from_utf8(&callable_name).unwrap();
|
|
match comp
|
|
.description()
|
|
.get_global(comp.borrow(), &normalize_identifier(std::str::from_utf8(&global).unwrap()))
|
|
.and_then(|g| {
|
|
if matches!(
|
|
comp.description()
|
|
.original
|
|
.root_element
|
|
.borrow()
|
|
.lookup_property(callable_name)
|
|
.property_type,
|
|
i_slint_compiler::langtype::Type::Function { .. }
|
|
) {
|
|
g.as_ref().eval_function(
|
|
&normalize_identifier(callable_name),
|
|
args.as_slice().iter().cloned().collect(),
|
|
)
|
|
} else {
|
|
g.as_ref().invoke_callback(&normalize_identifier(callable_name), args.as_slice())
|
|
}
|
|
}) {
|
|
Ok(val) => Box::into_raw(Box::new(val)),
|
|
Err(_) => std::ptr::null_mut(),
|
|
}
|
|
}
|
|
|
|
/// Show or hide
|
|
#[unsafe(no_mangle)]
|
|
pub extern "C" fn slint_interpreter_component_instance_show(
|
|
inst: &ErasedItemTreeBox,
|
|
is_visible: bool,
|
|
) {
|
|
generativity::make_guard!(guard);
|
|
let comp = inst.unerase(guard);
|
|
match is_visible {
|
|
true => comp.borrow_instance().window_adapter().window().show().unwrap(),
|
|
false => comp.borrow_instance().window_adapter().window().hide().unwrap(),
|
|
}
|
|
}
|
|
|
|
/// Return a window for the component
|
|
///
|
|
/// The out pointer must be uninitialized and must be destroyed with
|
|
/// slint_windowrc_drop after usage
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_instance_window(
|
|
inst: &ErasedItemTreeBox,
|
|
out: *mut *const i_slint_core::window::ffi::WindowAdapterRcOpaque,
|
|
) {
|
|
assert_eq!(
|
|
core::mem::size_of::<Rc<dyn WindowAdapter>>(),
|
|
core::mem::size_of::<i_slint_core::window::ffi::WindowAdapterRcOpaque>()
|
|
);
|
|
core::ptr::write(
|
|
out as *mut *const Rc<dyn WindowAdapter>,
|
|
inst.window_adapter_ref().unwrap() as *const _,
|
|
)
|
|
}
|
|
|
|
/// Instantiate an instance from a definition.
|
|
///
|
|
/// The `out` must be uninitialized and is going to be initialized after the call
|
|
/// and need to be destroyed with slint_interpreter_component_instance_destructor
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_instance_create(
|
|
def: &ComponentDefinitionOpaque,
|
|
out: *mut ComponentInstance,
|
|
) {
|
|
std::ptr::write(out, def.as_component_definition().create().unwrap())
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_instance_component_definition(
|
|
inst: &ErasedItemTreeBox,
|
|
component_definition_ptr: *mut ComponentDefinitionOpaque,
|
|
) {
|
|
generativity::make_guard!(guard);
|
|
let definition = ComponentDefinition { inner: inst.unerase(guard).description().into() };
|
|
std::ptr::write(component_definition_ptr as *mut ComponentDefinition, definition);
|
|
}
|
|
|
|
#[vtable::vtable]
|
|
#[repr(C)]
|
|
pub struct ModelAdaptorVTable {
|
|
pub row_count: extern "C" fn(VRef<ModelAdaptorVTable>) -> usize,
|
|
pub row_data: unsafe extern "C" fn(VRef<ModelAdaptorVTable>, row: usize) -> *mut Value,
|
|
pub set_row_data: extern "C" fn(VRef<ModelAdaptorVTable>, row: usize, value: Box<Value>),
|
|
pub get_notify: extern "C" fn(VRef<'_, ModelAdaptorVTable>) -> &ModelNotifyOpaque,
|
|
pub drop: extern "C" fn(VRefMut<ModelAdaptorVTable>),
|
|
}
|
|
|
|
struct ModelAdaptorWrapper(vtable::VBox<ModelAdaptorVTable>);
|
|
impl Model for ModelAdaptorWrapper {
|
|
type Data = Value;
|
|
|
|
fn row_count(&self) -> usize {
|
|
self.0.row_count()
|
|
}
|
|
|
|
fn row_data(&self, row: usize) -> Option<Value> {
|
|
let val_ptr = unsafe { self.0.row_data(row) };
|
|
if val_ptr.is_null() {
|
|
None
|
|
} else {
|
|
Some(*unsafe { Box::from_raw(val_ptr) })
|
|
}
|
|
}
|
|
|
|
fn model_tracker(&self) -> &dyn i_slint_core::model::ModelTracker {
|
|
self.0.get_notify().as_model_notify()
|
|
}
|
|
|
|
fn set_row_data(&self, row: usize, data: Value) {
|
|
let val = Box::new(data);
|
|
self.0.set_row_data(row, val);
|
|
}
|
|
}
|
|
|
|
#[repr(C)]
|
|
#[cfg(target_pointer_width = "64")]
|
|
pub struct ModelNotifyOpaque([usize; 8]);
|
|
#[repr(C)]
|
|
#[cfg(target_pointer_width = "32")]
|
|
pub struct ModelNotifyOpaque([usize; 12]);
|
|
/// Asserts that ModelNotifyOpaque is at least as large as ModelNotify, otherwise this would overflow
|
|
const _: usize = std::mem::size_of::<ModelNotifyOpaque>() - std::mem::size_of::<ModelNotify>();
|
|
const _: usize = std::mem::align_of::<ModelNotifyOpaque>() - std::mem::align_of::<ModelNotify>();
|
|
|
|
impl ModelNotifyOpaque {
|
|
fn as_model_notify(&self) -> &ModelNotify {
|
|
// Safety: there should be no way to construct a ModelNotifyOpaque without it holding an actual ModelNotify
|
|
unsafe { std::mem::transmute::<&ModelNotifyOpaque, &ModelNotify>(self) }
|
|
}
|
|
}
|
|
|
|
/// Construct a new ModelNotifyNotify in the given memory region
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_model_notify_new(val: *mut ModelNotifyOpaque) {
|
|
std::ptr::write(val as *mut ModelNotify, ModelNotify::default());
|
|
}
|
|
|
|
/// Destruct the value in that memory location
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_model_notify_destructor(val: *mut ModelNotifyOpaque) {
|
|
drop(std::ptr::read(val as *mut ModelNotify))
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_model_notify_row_changed(
|
|
notify: &ModelNotifyOpaque,
|
|
row: usize,
|
|
) {
|
|
notify.as_model_notify().row_changed(row);
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_model_notify_row_added(
|
|
notify: &ModelNotifyOpaque,
|
|
row: usize,
|
|
count: usize,
|
|
) {
|
|
notify.as_model_notify().row_added(row, count);
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_model_notify_reset(notify: &ModelNotifyOpaque) {
|
|
notify.as_model_notify().reset();
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_model_notify_row_removed(
|
|
notify: &ModelNotifyOpaque,
|
|
row: usize,
|
|
count: usize,
|
|
) {
|
|
notify.as_model_notify().row_removed(row, count);
|
|
}
|
|
|
|
// FIXME: Figure out how to re-export the one from compilerlib
|
|
/// DiagnosticLevel describes the severity of a diagnostic.
|
|
#[derive(Clone)]
|
|
#[repr(u8)]
|
|
pub enum DiagnosticLevel {
|
|
/// The diagnostic belongs to an error.
|
|
Error,
|
|
/// The diagnostic belongs to a warning.
|
|
Warning,
|
|
}
|
|
|
|
/// Diagnostic describes the aspects of either a warning or an error, along
|
|
/// with its location and a description. Diagnostics are typically returned by
|
|
/// slint::interpreter::ComponentCompiler::diagnostics() in a vector.
|
|
#[derive(Clone)]
|
|
#[repr(C)]
|
|
pub struct Diagnostic {
|
|
/// The message describing the warning or error.
|
|
message: SharedString,
|
|
/// The path to the source file where the warning or error is located.
|
|
source_file: SharedString,
|
|
/// The line within the source file. Line numbers start at 1.
|
|
line: usize,
|
|
/// The column within the source file. Column numbers start at 1.
|
|
column: usize,
|
|
/// The level of the diagnostic, such as a warning or an error.
|
|
level: DiagnosticLevel,
|
|
}
|
|
|
|
#[repr(transparent)]
|
|
pub struct ComponentCompilerOpaque(#[allow(deprecated)] NonNull<ComponentCompiler>);
|
|
|
|
#[allow(deprecated)]
|
|
impl ComponentCompilerOpaque {
|
|
fn as_component_compiler(&self) -> &ComponentCompiler {
|
|
// Safety: there should be no way to construct a ComponentCompilerOpaque without it holding an actual ComponentCompiler
|
|
unsafe { self.0.as_ref() }
|
|
}
|
|
fn as_component_compiler_mut(&mut self) -> &mut ComponentCompiler {
|
|
// Safety: there should be no way to construct a ComponentCompilerOpaque without it holding an actual ComponentCompiler
|
|
unsafe { self.0.as_mut() }
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
#[allow(deprecated)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_compiler_new(
|
|
compiler: *mut ComponentCompilerOpaque,
|
|
) {
|
|
*compiler = ComponentCompilerOpaque(NonNull::new_unchecked(Box::into_raw(Box::new(
|
|
ComponentCompiler::default(),
|
|
))));
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_compiler_destructor(
|
|
compiler: *mut ComponentCompilerOpaque,
|
|
) {
|
|
drop(Box::from_raw((*compiler).0.as_ptr()))
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_compiler_set_include_paths(
|
|
compiler: &mut ComponentCompilerOpaque,
|
|
paths: &SharedVector<SharedString>,
|
|
) {
|
|
compiler
|
|
.as_component_compiler_mut()
|
|
.set_include_paths(paths.iter().map(|path| path.as_str().into()).collect())
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_compiler_set_style(
|
|
compiler: &mut ComponentCompilerOpaque,
|
|
style: Slice<u8>,
|
|
) {
|
|
compiler.as_component_compiler_mut().set_style(std::str::from_utf8(&style).unwrap().to_string())
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_compiler_set_translation_domain(
|
|
compiler: &mut ComponentCompilerOpaque,
|
|
translation_domain: Slice<u8>,
|
|
) {
|
|
compiler
|
|
.as_component_compiler_mut()
|
|
.set_translation_domain(std::str::from_utf8(&translation_domain).unwrap().to_string())
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_compiler_get_style(
|
|
compiler: &ComponentCompilerOpaque,
|
|
style_out: &mut SharedString,
|
|
) {
|
|
*style_out =
|
|
compiler.as_component_compiler().style().map_or(SharedString::default(), |s| s.into());
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_compiler_get_include_paths(
|
|
compiler: &ComponentCompilerOpaque,
|
|
paths: &mut SharedVector<SharedString>,
|
|
) {
|
|
paths.extend(
|
|
compiler
|
|
.as_component_compiler()
|
|
.include_paths()
|
|
.iter()
|
|
.map(|path| path.to_str().map_or_else(Default::default, |str| str.into())),
|
|
);
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_compiler_get_diagnostics(
|
|
compiler: &ComponentCompilerOpaque,
|
|
out_diags: &mut SharedVector<Diagnostic>,
|
|
) {
|
|
#[allow(deprecated)]
|
|
out_diags.extend(compiler.as_component_compiler().diagnostics.iter().map(|diagnostic| {
|
|
let (line, column) = diagnostic.line_column();
|
|
Diagnostic {
|
|
message: diagnostic.message().into(),
|
|
source_file: diagnostic
|
|
.source_file()
|
|
.and_then(|path| path.to_str())
|
|
.map_or_else(Default::default, |str| str.into()),
|
|
line,
|
|
column,
|
|
level: match diagnostic.level() {
|
|
i_slint_compiler::diagnostics::DiagnosticLevel::Error => DiagnosticLevel::Error,
|
|
i_slint_compiler::diagnostics::DiagnosticLevel::Warning => DiagnosticLevel::Warning,
|
|
_ => DiagnosticLevel::Warning,
|
|
},
|
|
}
|
|
}));
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_compiler_build_from_source(
|
|
compiler: &mut ComponentCompilerOpaque,
|
|
source_code: Slice<u8>,
|
|
path: Slice<u8>,
|
|
component_definition_ptr: *mut ComponentDefinitionOpaque,
|
|
) -> bool {
|
|
match spin_on::spin_on(compiler.as_component_compiler_mut().build_from_source(
|
|
std::str::from_utf8(&source_code).unwrap().to_string(),
|
|
std::str::from_utf8(&path).unwrap().to_string().into(),
|
|
)) {
|
|
Some(definition) => {
|
|
std::ptr::write(component_definition_ptr as *mut ComponentDefinition, definition);
|
|
true
|
|
}
|
|
None => false,
|
|
}
|
|
}
|
|
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_compiler_build_from_path(
|
|
compiler: &mut ComponentCompilerOpaque,
|
|
path: Slice<u8>,
|
|
component_definition_ptr: *mut ComponentDefinitionOpaque,
|
|
) -> bool {
|
|
use std::str::FromStr;
|
|
match spin_on::spin_on(
|
|
compiler
|
|
.as_component_compiler_mut()
|
|
.build_from_path(PathBuf::from_str(std::str::from_utf8(&path).unwrap()).unwrap()),
|
|
) {
|
|
Some(definition) => {
|
|
std::ptr::write(component_definition_ptr as *mut ComponentDefinition, definition);
|
|
true
|
|
}
|
|
None => false,
|
|
}
|
|
}
|
|
|
|
/// PropertyDescriptor is a simple structure that's used to describe a property declared in .slint
|
|
/// code. It is returned from in a vector from
|
|
/// slint::interpreter::ComponentDefinition::properties().
|
|
#[derive(Clone)]
|
|
#[repr(C)]
|
|
pub struct PropertyDescriptor {
|
|
/// The name of the declared property.
|
|
property_name: SharedString,
|
|
/// The type of the property.
|
|
property_type: ValueType,
|
|
}
|
|
|
|
#[repr(C)]
|
|
// Note: This needs to stay the size of 1 pointer to allow for the null pointer definition
|
|
// in the C++ wrapper to allow for the null state.
|
|
pub struct ComponentDefinitionOpaque([usize; 1]);
|
|
/// Asserts that ComponentCompilerOpaque is as large as ComponentCompiler and has the same alignment, to make transmute safe.
|
|
const _: [(); std::mem::size_of::<ComponentDefinitionOpaque>()] =
|
|
[(); std::mem::size_of::<ComponentDefinition>()];
|
|
const _: [(); std::mem::align_of::<ComponentDefinitionOpaque>()] =
|
|
[(); std::mem::align_of::<ComponentDefinition>()];
|
|
|
|
impl ComponentDefinitionOpaque {
|
|
fn as_component_definition(&self) -> &ComponentDefinition {
|
|
// Safety: there should be no way to construct a ComponentDefinitionOpaque without it holding an actual ComponentDefinition
|
|
unsafe { std::mem::transmute::<&ComponentDefinitionOpaque, &ComponentDefinition>(self) }
|
|
}
|
|
}
|
|
|
|
/// Construct a new Value in the given memory location
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_definition_clone(
|
|
other: &ComponentDefinitionOpaque,
|
|
def: *mut ComponentDefinitionOpaque,
|
|
) {
|
|
std::ptr::write(def as *mut ComponentDefinition, other.as_component_definition().clone())
|
|
}
|
|
|
|
/// Destruct the component definition in that memory location
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_definition_destructor(
|
|
val: *mut ComponentDefinitionOpaque,
|
|
) {
|
|
drop(std::ptr::read(val as *mut ComponentDefinition))
|
|
}
|
|
|
|
/// Returns the list of properties of the component the component definition describes
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_definition_properties(
|
|
def: &ComponentDefinitionOpaque,
|
|
props: &mut SharedVector<PropertyDescriptor>,
|
|
) {
|
|
props.extend((&*def).as_component_definition().properties().map(
|
|
|(property_name, property_type)| PropertyDescriptor {
|
|
property_name: property_name.into(),
|
|
property_type,
|
|
},
|
|
))
|
|
}
|
|
|
|
/// Returns the list of callback names of the component the component definition describes
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_definition_callbacks(
|
|
def: &ComponentDefinitionOpaque,
|
|
callbacks: &mut SharedVector<SharedString>,
|
|
) {
|
|
callbacks.extend((&*def).as_component_definition().callbacks().map(|name| name.into()))
|
|
}
|
|
|
|
/// Returns the list of function names of the component the component definition describes
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_definition_functions(
|
|
def: &ComponentDefinitionOpaque,
|
|
functions: &mut SharedVector<SharedString>,
|
|
) {
|
|
functions.extend((&*def).as_component_definition().functions().map(|name| name.into()))
|
|
}
|
|
|
|
/// Return the name of the component definition
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_definition_name(
|
|
def: &ComponentDefinitionOpaque,
|
|
name: &mut SharedString,
|
|
) {
|
|
*name = (&*def).as_component_definition().name().into()
|
|
}
|
|
|
|
/// Returns a vector of strings with the names of all exported global singletons.
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_definition_globals(
|
|
def: &ComponentDefinitionOpaque,
|
|
names: &mut SharedVector<SharedString>,
|
|
) {
|
|
names.extend((&*def).as_component_definition().globals().map(|name| name.into()))
|
|
}
|
|
|
|
/// Returns a vector of the property descriptors of the properties of the specified publicly exported global
|
|
/// singleton. Returns true if a global exists under the specified name; false otherwise.
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_definition_global_properties(
|
|
def: &ComponentDefinitionOpaque,
|
|
global_name: Slice<u8>,
|
|
properties: &mut SharedVector<PropertyDescriptor>,
|
|
) -> bool {
|
|
if let Some(property_it) = (&*def)
|
|
.as_component_definition()
|
|
.global_properties(std::str::from_utf8(&global_name).unwrap())
|
|
{
|
|
properties.extend(property_it.map(|(property_name, property_type)| PropertyDescriptor {
|
|
property_name: property_name.into(),
|
|
property_type,
|
|
}));
|
|
true
|
|
} else {
|
|
false
|
|
}
|
|
}
|
|
|
|
/// Returns a vector of the names of the callbacks of the specified publicly exported global
|
|
/// singleton. Returns true if a global exists under the specified name; false otherwise.
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_definition_global_callbacks(
|
|
def: &ComponentDefinitionOpaque,
|
|
global_name: Slice<u8>,
|
|
names: &mut SharedVector<SharedString>,
|
|
) -> bool {
|
|
if let Some(name_it) = (&*def)
|
|
.as_component_definition()
|
|
.global_callbacks(std::str::from_utf8(&global_name).unwrap())
|
|
{
|
|
names.extend(name_it.map(|name| name.into()));
|
|
true
|
|
} else {
|
|
false
|
|
}
|
|
}
|
|
|
|
/// Returns a vector of the names of the functions of the specified publicly exported global
|
|
/// singleton. Returns true if a global exists under the specified name; false otherwise.
|
|
#[unsafe(no_mangle)]
|
|
pub unsafe extern "C" fn slint_interpreter_component_definition_global_functions(
|
|
def: &ComponentDefinitionOpaque,
|
|
global_name: Slice<u8>,
|
|
names: &mut SharedVector<SharedString>,
|
|
) -> bool {
|
|
if let Some(name_it) = (&*def)
|
|
.as_component_definition()
|
|
.global_functions(std::str::from_utf8(&global_name).unwrap())
|
|
{
|
|
names.extend(name_it.map(|name| name.into()));
|
|
true
|
|
} else {
|
|
false
|
|
}
|
|
}
|