use adw::{prelude::*, subclass::prelude::*};
use gettextrs::gettext;
use gtk::{glib, glib::clone, CompositeTemplate};
use crate::{session::model::Room, toast, utils::BoundObjectWeakRef, Window};
mod imp {
use glib::subclass::InitializingObject;
use super::*;
#[derive(Debug, Default, CompositeTemplate, glib::Properties)]
#[template(
resource = "/org/gnome/Fractal/ui/session/view/content/room_history/state_row/tombstone.ui"
)]
#[properties(wrapper_type = super::StateTombstone)]
pub struct StateTombstone {
#[template_child]
pub new_room_btn: TemplateChild<gtk::Button>,
#[property(get, set = Self::set_room, construct_only)]
pub room: BoundObjectWeakRef<Room>,
}
#[glib::object_subclass]
impl ObjectSubclass for StateTombstone {
const NAME: &'static str = "ContentStateTombstone";
type Type = super::StateTombstone;
type ParentType = adw::Bin;
fn class_init(klass: &mut Self::Class) {
Self::bind_template(klass);
Self::Type::bind_template_callbacks(klass);
}
fn instance_init(obj: &InitializingObject<Self>) {
obj.init_template();
}
}
#[glib::derived_properties]
impl ObjectImpl for StateTombstone {}
impl WidgetImpl for StateTombstone {}
impl BinImpl for StateTombstone {}
impl StateTombstone {
fn set_room(&self, room: Room) {
let obj = self.obj();
let successor_handler = room.connect_successor_id_string_notify(clone!(
#[weak(rename_to = imp)]
self,
move |room| {
imp.new_room_btn.set_visible(room.successor_id().is_some());
}
));
self.new_room_btn.set_visible(room.successor_id().is_some());
let successor_room_handler = room.connect_successor_notify(clone!(
#[weak]
obj,
move |room| {
obj.update_button_label(room);
}
));
obj.update_button_label(&room);
self.room
.set(&room, vec![successor_handler, successor_room_handler]);
}
}
}
glib::wrapper! {
pub struct StateTombstone(ObjectSubclass<imp::StateTombstone>)
@extends gtk::Widget, adw::Bin, @implements gtk::Accessible;
}
#[gtk::template_callbacks]
impl StateTombstone {
pub fn new(room: &Room) -> Self {
glib::Object::builder().property("room", room).build()
}
fn update_button_label(&self, room: &Room) {
let button = &self.imp().new_room_btn;
if room.successor().is_some() {
button.set_label(&gettext("View"));
} else {
button.set_label(&gettext("Join"));
}
}
#[template_callback]
async fn join_or_view_successor(&self) {
let Some(room) = self.room() else {
return;
};
let Some(session) = room.session() else {
return;
};
let room_list = session.room_list();
if let Some(successor) = room.successor() {
let Some(window) = self.root().and_downcast::<Window>() else {
return;
};
window.session_view().select_room(Some(successor));
} else if let Some(successor_id) = room.successor_id().map(ToOwned::to_owned) {
let via = successor_id
.server_name()
.map(ToOwned::to_owned)
.into_iter()
.collect();
if let Err(error) = room_list
.join_by_id_or_alias(successor_id.into(), via)
.await
{
toast!(self, error);
}
}
}
}