use std::sync::OnceLock;
use glib::{prelude::*, subclass::prelude::*, translate::*};
use crate::{ffi, ListModel};
pub trait ListModelImpl: ObjectImpl {
#[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>;
}
mod sealed {
pub trait Sealed {}
impl<T: super::ListModelImplExt> Sealed for T {}
}
pub trait ListModelImplExt: sealed::Sealed + ObjectSubclass {
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
where
<T as ObjectSubclass>::Type: IsA<glib::Object>,
{
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
where
<T as ObjectSubclass>::Type: IsA<glib::Object>,
{
let instance = &*(list_model as *mut T::Instance);
let imp = instance.imp();
let type_ = imp.item_type().into_glib();
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
where
<T as ObjectSubclass>::Type: IsA<glib::Object>,
{
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
where
<T as ObjectSubclass>::Type: IsA<glib::Object>,
{
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()
}