use gtk::{
gio, glib,
glib::{clone, closure},
prelude::*,
subclass::prelude::*,
};
mod category_filter;
mod category_type;
use self::category_filter::CategoryFilter;
pub use self::category_type::CategoryType;
use crate::{
session::model::{Room, RoomList, RoomType, SessionSettings, VerificationList},
utils::ExpressionListModel,
};
mod imp {
use std::{
cell::{Cell, OnceCell},
marker::PhantomData,
};
use super::*;
#[derive(Debug, Default, glib::Properties)]
#[properties(wrapper_type = super::Category)]
pub struct Category {
#[property(get, set = Self::set_model, construct_only)]
pub model: OnceCell<gio::ListModel>,
inner_model: OnceCell<gio::ListModel>,
pub filter: CategoryFilter,
#[property(get = Self::category_type, set = Self::set_category_type, construct_only, builder(CategoryType::default()))]
pub category_type: PhantomData<CategoryType>,
#[property(get)]
pub empty: Cell<bool>,
#[property(get = Self::display_name)]
pub display_name: PhantomData<String>,
#[property(get, set = Self::set_is_expanded, explicit_notify)]
pub is_expanded: Cell<bool>,
}
#[glib::object_subclass]
impl ObjectSubclass for Category {
const NAME: &'static str = "Category";
type Type = super::Category;
type Interfaces = (gio::ListModel,);
}
#[glib::derived_properties]
impl ObjectImpl for Category {
fn constructed(&self) {
self.parent_constructed();
let Some(settings) = self.session_settings() else {
return;
};
let is_expanded = settings.is_category_expanded(self.category_type());
self.set_is_expanded(is_expanded);
}
}
impl ListModelImpl for Category {
fn item_type(&self) -> glib::Type {
glib::Object::static_type()
}
fn n_items(&self) -> u32 {
self.inner_model().n_items()
}
fn item(&self, position: u32) -> Option<glib::Object> {
self.inner_model().item(position)
}
}
impl Category {
fn model(&self) -> &gio::ListModel {
self.model.get().unwrap()
}
fn set_model(&self, model: gio::ListModel) {
let model = self.model.get_or_init(|| model).clone();
let obj = self.obj();
let inner_model = if model.is::<RoomList>() {
let room_category_type = Room::this_expression("category")
.chain_closure::<CategoryType>(closure!(
|_: Option<glib::Object>, room_type: RoomType| {
CategoryType::from(room_type)
}
));
self.filter
.set_expression(Some(room_category_type.clone().upcast()));
let category_type_expr_model = ExpressionListModel::new();
category_type_expr_model.set_expressions(vec![room_category_type.upcast()]);
category_type_expr_model.set_model(Some(model));
let filter_model = gtk::FilterListModel::new(
Some(category_type_expr_model),
Some(self.filter.clone()),
);
let room_latest_activity = Room::this_expression("latest-activity");
let sorter = gtk::NumericSorter::builder()
.expression(&room_latest_activity)
.sort_order(gtk::SortType::Descending)
.build();
let latest_activity_expr_model = ExpressionListModel::new();
latest_activity_expr_model.set_expressions(vec![room_latest_activity.upcast()]);
latest_activity_expr_model.set_model(Some(filter_model));
let sort_model =
gtk::SortListModel::new(Some(latest_activity_expr_model), Some(sorter));
sort_model.upcast()
} else {
model
};
inner_model.connect_items_changed(clone!(
#[weak]
obj,
move |model, pos, removed, added| {
obj.items_changed(pos, removed, added);
obj.imp().set_empty(model.n_items() == 0);
}
));
self.set_empty(inner_model.n_items() == 0);
self.inner_model.set(inner_model).unwrap();
}
fn inner_model(&self) -> &gio::ListModel {
self.inner_model.get().unwrap()
}
fn category_type(&self) -> CategoryType {
self.filter.category_type()
}
fn set_category_type(&self, type_: CategoryType) {
self.filter.set_category_type(type_);
}
fn set_empty(&self, empty: bool) {
if empty == self.empty.get() {
return;
}
self.empty.set(empty);
self.obj().notify_empty();
}
fn display_name(&self) -> String {
self.category_type().to_string()
}
fn set_is_expanded(&self, expanded: bool) {
if self.is_expanded.get() == expanded {
return;
}
self.is_expanded.set(expanded);
self.obj().notify_is_expanded();
if let Some(settings) = self.session_settings() {
settings.set_category_expanded(self.category_type(), expanded);
}
}
fn session_settings(&self) -> Option<SessionSettings> {
let model = self.model();
let session = model
.downcast_ref::<RoomList>()
.and_then(|l| l.session())
.or_else(|| {
model
.downcast_ref::<VerificationList>()
.and_then(|l| l.session())
})?;
Some(session.settings())
}
}
}
glib::wrapper! {
pub struct Category(ObjectSubclass<imp::Category>)
@implements gio::ListModel;
}
impl Category {
pub fn new(category_type: CategoryType, model: &impl IsA<gio::ListModel>) -> Self {
glib::Object::builder()
.property("category-type", category_type)
.property("model", model)
.build()
}
pub fn visible_for_category(&self, for_category: CategoryType) -> bool {
if !self.empty() {
return true;
}
let room_types = RoomType::try_from(for_category)
.ok()
.zip(RoomType::try_from(self.category_type()).ok());
room_types.is_some_and(|(source_room_type, target_room_type)| {
source_room_type.can_change_to(target_room_type)
})
}
}