fractal/src/components/loading_listbox_row.rs

174 lines
5.1 KiB
Rust

use glib::subclass::Signal;
use gtk::{glib, glib::clone, prelude::*, subclass::prelude::*, CompositeTemplate};
use super::Spinner;
mod imp {
use std::cell::Cell;
use glib::subclass::InitializingObject;
use once_cell::sync::Lazy;
use super::*;
#[derive(Debug, Default, CompositeTemplate)]
#[template(resource = "/org/gnome/Fractal/ui/components/loading_listbox_row.ui")]
pub struct LoadingListBoxRow {
#[template_child]
pub spinner: TemplateChild<Spinner>,
#[template_child]
pub stack: TemplateChild<gtk::Stack>,
#[template_child]
pub error: TemplateChild<gtk::Box>,
#[template_child]
pub error_label: TemplateChild<gtk::Label>,
#[template_child]
pub retry_button: TemplateChild<gtk::Button>,
pub is_error: Cell<bool>,
}
#[glib::object_subclass]
impl ObjectSubclass for LoadingListBoxRow {
const NAME: &'static str = "ComponentsLoadingListBoxRow";
type Type = super::LoadingListBoxRow;
type ParentType = gtk::ListBoxRow;
fn class_init(klass: &mut Self::Class) {
Self::bind_template(klass);
}
fn instance_init(obj: &InitializingObject<Self>) {
obj.init_template();
}
}
impl ObjectImpl for LoadingListBoxRow {
fn properties() -> &'static [glib::ParamSpec] {
static PROPERTIES: Lazy<Vec<glib::ParamSpec>> = Lazy::new(|| {
vec![
glib::ParamSpecBoolean::builder("loading")
.default_value(true)
.explicit_notify()
.build(),
glib::ParamSpecString::builder("error")
.explicit_notify()
.build(),
]
});
PROPERTIES.as_ref()
}
fn set_property(&self, _id: usize, value: &glib::Value, pspec: &glib::ParamSpec) {
let obj = self.obj();
match pspec.name() {
"loading" => {
obj.set_loading(value.get().unwrap());
}
"error" => {
obj.set_error(value.get().unwrap());
}
_ => unimplemented!(),
}
}
fn property(&self, _id: usize, pspec: &glib::ParamSpec) -> glib::Value {
let obj = self.obj();
match pspec.name() {
"loading" => obj.is_loading().to_value(),
"error" => obj.error().to_value(),
_ => unimplemented!(),
}
}
fn signals() -> &'static [Signal] {
static SIGNALS: Lazy<Vec<Signal>> =
Lazy::new(|| vec![Signal::builder("retry").build()]);
SIGNALS.as_ref()
}
fn constructed(&self) {
self.parent_constructed();
let obj = self.obj();
self.retry_button
.connect_clicked(clone!(@weak obj => move |_| {
obj.emit_by_name::<()>("retry", &[]);
}));
}
}
impl WidgetImpl for LoadingListBoxRow {}
impl ListBoxRowImpl for LoadingListBoxRow {}
}
glib::wrapper! {
/// This is a `ListBoxRow` containing a loading spinner.
///
/// It's also possible to set an error once the loading fails including a retry button.
pub struct LoadingListBoxRow(ObjectSubclass<imp::LoadingListBoxRow>)
@extends gtk::Widget, gtk::ListBoxRow, @implements gtk::Accessible;
}
impl LoadingListBoxRow {
pub fn new() -> Self {
glib::Object::new()
}
/// Whether to show the loading spinner.
pub fn is_loading(&self) -> bool {
!self.imp().is_error.get()
}
/// Set whether to show the loading spinner.
pub fn set_loading(&self, loading: bool) {
let imp = self.imp();
if self.is_loading() == loading {
return;
}
imp.stack.set_visible_child(&*imp.spinner);
imp.is_error.set(false);
self.notify("loading");
}
/// The error message to display.
pub fn error(&self) -> Option<glib::GString> {
let message = self.imp().error_label.text();
if message.is_empty() {
None
} else {
Some(message)
}
}
/// Set the error message to display.
///
/// If this is `Some`, the error will be shown, otherwise the spinner will
/// be shown.
pub fn set_error(&self, message: Option<&str>) {
let imp = self.imp();
if let Some(message) = message {
imp.is_error.set(true);
imp.error_label.set_text(message);
imp.stack.set_visible_child(&*imp.error);
} else {
imp.is_error.set(false);
imp.stack.set_visible_child(&*imp.spinner);
}
self.notify("error");
}
pub fn connect_retry<F: Fn(&Self) + 'static>(&self, f: F) -> glib::SignalHandlerId {
self.connect_local("retry", true, move |values| {
let obj = values[0].get::<Self>().unwrap();
f(&obj);
None
})
}
}