【CXX-Qt】2.4 嵌套对象

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

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

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

rust 复制代码
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>`。
rust 复制代码
#[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();
    }
}
相关推荐
pumpkin845141 小时前
Rust Mock 工具
开发语言·rust
kingmax542120083 小时前
【洛谷P9303题解】AC- [CCC 2023 J5] CCC Word Hunt
数据结构·c++·算法·广度优先
AgilityBaby5 小时前
UE5打包项目设置Project Settings(打包widows exe安装包)
c++·3d·ue5·游戏引擎·unreal engine
cykaw25906 小时前
QT-JSON
qt·json
UestcXiye7 小时前
Rust 学习笔记:发布一个 crate 到 crates.io
rust
让我们一起加油好吗7 小时前
【基础算法】高精度(加、减、乘、除)
c++·算法·高精度·洛谷
Forest_HAHA7 小时前
<5>, Qt系统相关
开发语言·qt
鑫鑫向栄8 小时前
[蓝桥杯]缩位求和
数据结构·c++·算法·职场和发展·蓝桥杯
stormsha8 小时前
MCP架构全解析:从核心原理到企业级实践
服务器·c++·架构
梁下轻语的秋缘8 小时前
每日c/c++题 备战蓝桥杯(P1204 [USACO1.2] 挤牛奶 Milking Cows)
c语言·c++·蓝桥杯