gio 0.22.2

Rust bindings for the Gio library
Documentation
// Take a look at the license at the top of the repository in the LICENSE file.

use std::sync::OnceLock;

use glib::{prelude::*, subclass::prelude::*, translate::*};

use crate::{ListModel, ffi};

pub trait ListModelImpl: ObjectImpl + ObjectSubclass<Type: IsA<ListModel>> {
    #[doc(alias = "get_item_type")]
    fn item_type(&self) -> glib::Type;
    #[doc(alias = "get_n_items")]
    fn n_items(&self) -> u32;
    #[doc(alias = "get_item")]
    fn item(&self, position: u32) -> Option<glib::Object>;
}

pub trait ListModelImplExt: ListModelImpl {
    fn parent_item_type(&self) -> glib::Type {
        unsafe {
            let type_data = Self::type_data();
            let parent_iface = type_data.as_ref().parent_interface::<ListModel>()
                as *const ffi::GListModelInterface;

            let func = (*parent_iface)
                .get_item_type
                .expect("no parent \"item_type\" implementation");
            let ret = func(self.obj().unsafe_cast_ref::<ListModel>().to_glib_none().0);
            from_glib(ret)
        }
    }

    fn parent_n_items(&self) -> u32 {
        unsafe {
            let type_data = Self::type_data();
            let parent_iface = type_data.as_ref().parent_interface::<ListModel>()
                as *const ffi::GListModelInterface;

            let func = (*parent_iface)
                .get_n_items
                .expect("no parent \"n_items\" implementation");
            func(self.obj().unsafe_cast_ref::<ListModel>().to_glib_none().0)
        }
    }

    fn parent_item(&self, position: u32) -> Option<glib::Object> {
        unsafe {
            let type_data = Self::type_data();
            let parent_iface = type_data.as_ref().parent_interface::<ListModel>()
                as *const ffi::GListModelInterface;

            let func = (*parent_iface)
                .get_item
                .expect("no parent \"get_item\" implementation");
            let ret = func(
                self.obj().unsafe_cast_ref::<ListModel>().to_glib_none().0,
                position,
            );
            from_glib_full(ret)
        }
    }
}

impl<T: ListModelImpl> ListModelImplExt for T {}

unsafe impl<T: ListModelImpl> IsImplementable<T> for ListModel {
    fn interface_init(iface: &mut glib::Interface<Self>) {
        let iface = iface.as_mut();

        iface.get_item_type = Some(list_model_get_item_type::<T>);
        iface.get_n_items = Some(list_model_get_n_items::<T>);
        iface.get_item = Some(list_model_get_item::<T>);
    }
}

unsafe extern "C" fn list_model_get_item_type<T: ListModelImpl>(
    list_model: *mut ffi::GListModel,
) -> glib::ffi::GType {
    unsafe {
        let instance = &*(list_model as *mut T::Instance);
        let imp = instance.imp();

        let type_ = imp.item_type().into_glib();

        // Store the type so we can enforce that it doesn't change.
        let instance = imp.obj();
        let type_quark = {
            static QUARK: OnceLock<glib::Quark> = OnceLock::new();
            *QUARK.get_or_init(|| glib::Quark::from_str("gtk-rs-subclass-list-model-item-type"))
        };
        match instance.qdata(type_quark) {
            Some(old_type) => {
                assert_eq!(
                    type_,
                    *old_type.as_ref(),
                    "ListModel's get_item_type cannot be changed"
                );
            }
            None => {
                instance.set_qdata(type_quark, type_);
            }
        }
        type_
    }
}

unsafe extern "C" fn list_model_get_n_items<T: ListModelImpl>(
    list_model: *mut ffi::GListModel,
) -> u32 {
    unsafe {
        let instance = &*(list_model as *mut T::Instance);
        let imp = instance.imp();

        imp.n_items()
    }
}

unsafe extern "C" fn list_model_get_item<T: ListModelImpl>(
    list_model: *mut ffi::GListModel,
    position: u32,
) -> *mut glib::gobject_ffi::GObject {
    unsafe {
        let instance = &*(list_model as *mut T::Instance);
        let imp = instance.imp();

        let item = imp.item(position);

        if let Some(ref i) = item {
            let type_ = imp.item_type();
            assert!(
                i.type_().is_a(type_),
                "All ListModel items need to be of type {} or a subtype of it",
                type_.name()
            );
        };
        item.into_glib_ptr()
    }
}