【CXX-Qt】2.4 嵌套对象

Rust Qt 对象可以作为属性或参数相互嵌套。

嵌套对象通过指向其 QObject 表示的指针来引用。

首先,在桥接模块的 extern 块中正常定义一个类型。

extern "RustQt" {
    #[qobject]
    #[qml_element]
    #[qproperty(i32, counter)]
    type InnerObject = super::InnerObjectRust;
}

然后,可以通过使用 *mut T 将其作为属性、可调用方法的参数或信号参数使用。如下例所示,将 InnerObject 嵌套到 OuterObject 中。

需要使用 C++ CXX 类型作为 `T` 类型,而不是 Rust 结构体。
要访问嵌套对象的可变可调用方法和属性设置器,需要将 `*mut T` 转换为 `Pin<&mut T>`。
#[cxx_qt::bridge]
pub mod qobject {
    extern "RustQt" {
        #[qobject]
        #[qml_element]
        #[qproperty(i32, counter)]
        type InnerObject = super::InnerObjectRust;
    }

    extern "RustQt" {
        /// 一个信号,展示如何将另一个 QObject 作为参数引用
        ///
        /// # 安全性
        ///
        /// 由于使用了裸指针,这在 CXX 中被视为不安全
        #[qsignal]
        unsafe fn called(self: Pin<&mut InnerObject>, inner: *mut InnerObject);
    }

    extern "RustQt" {
        #[qobject]
        #[qml_element]
        #[qproperty(*mut InnerObject, inner)]
        type OuterObject = super::OuterObjectRust;

        /// 一个信号,展示如何将另一个 QObject 作为参数引用
        ///
        /// # 安全性
        ///
        /// 由于使用了裸指针,这在 CXX 中被视为不安全
        #[qsignal]
        unsafe fn called(self: Pin<&mut OuterObject>, inner: *mut InnerObject);
    }

    unsafe extern "RustQt" {
        /// 打印给定内部 QObject 的计数
        ///
        /// # 安全性
        ///
        /// 由于在公共方法中解引用指针,这需要标记为不安全
        #[qinvokable]
        #[cxx_name = "printCount"]
        unsafe fn print_count(self: Pin<&mut OuterObject>, inner: *mut InnerObject);

        /// 重置存储在 Q_PROPERTY 中的内部 QObject 的计数器
        #[qinvokable]
        fn reset(self: Pin<&mut OuterObject>);
    }

    impl cxx_qt::Constructor<()> for OuterObject {}
}

use core::pin::Pin;

/// 内部 QObject
#[derive(Default)]
pub struct InnerObjectRust {
    counter: i32,
}

/// 外部 QObject,具有指向内部 QObject 的 Q_PROPERTY
pub struct OuterObjectRust {
    inner: *mut qobject::InnerObject,
}

impl Default for OuterObjectRust {
    fn default() -> Self {
        Self {
            inner: std::ptr::null_mut(),
        }
    }
}

impl qobject::OuterObject {
    /// 打印给定内部 QObject 的计数
    ///
    /// # 安全性
    ///
    /// 由于在公共方法中解引用指针,这需要标记为不安全
    pub unsafe fn print_count(self: Pin<&mut Self>, inner: *mut qobject::InnerObject) {
        if let Some(inner) = inner.as_ref() {
            println!("内部对象的计数器属性: {}", inner.counter());
        }

        self.called(inner);
    }

    /// 重置存储在 Q_PROPERTY 中的内部 QObject 的计数器
    pub fn reset(self: Pin<&mut Self>) {
        // 需要将 *mut T 转换为 Pin<&mut T>,以便访问方法
        if let Some(inner) = unsafe { self.inner().as_mut() } {
            let pinned_inner = unsafe { Pin::new_unchecked(inner) };
            // 现在可以像平常一样使用 pinned_inner
            pinned_inner.set_counter(10);
        }

        // 获取 *mut T
        let inner = *self.inner();
        unsafe { self.called(inner) };
    }
}

impl cxx_qt::Initialize for qobject::OuterObject {
    /// 初始化 QObject,将一个信号连接到另一个信号
    fn initialize(self: core::pin::Pin<&mut Self>) {
        // 示例:将一个 QObject 的信号连接到另一个 QObject 的信号
        // 这将触发 OuterObject::Called 以触发 InnerObject::Called
        self.on_called(|qobject, obj| {
            // 需要将 *mut T 转换为 Pin<&mut T>,以便访问方法
            if let Some(inner) = unsafe { qobject.inner().as_mut() } {
                let pinned_inner = unsafe { Pin::new_unchecked(inner) };
                // 现在可以像平常一样使用 pinned_inner
                unsafe {
                    pinned_inner.called(obj);
                }
            }
        })
        .release();
    }
}
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

liuyuan77

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值