use adw::subclass::prelude::*;
use gtk::{glib, glib::closure_local, prelude::*, CompositeTemplate};
use super::Spinner;
#[derive(Debug, Default, Hash, Eq, PartialEq, Clone, Copy, glib::Enum)]
#[repr(u32)]
#[enum_type(name = "ActionState")]
pub enum ActionState {
#[default]
Default = 0,
Confirm = 1,
Retry = 2,
Loading = 3,
Success = 4,
Warning = 5,
Error = 6,
}
impl AsRef<str> for ActionState {
fn as_ref(&self) -> &str {
match self {
ActionState::Default => "default",
ActionState::Confirm => "confirm",
ActionState::Retry => "retry",
ActionState::Loading => "loading",
ActionState::Success => "success",
ActionState::Warning => "warning",
ActionState::Error => "error",
}
}
}
mod imp {
use std::{
cell::{Cell, RefCell},
marker::PhantomData,
};
use glib::subclass::{InitializingObject, Signal};
use once_cell::sync::Lazy;
use super::*;
#[derive(Debug, Default, CompositeTemplate, glib::Properties)]
#[template(resource = "/org/gnome/Fractal/ui/components/action_button.ui")]
#[properties(wrapper_type = super::ActionButton)]
pub struct ActionButton {
#[property(get, set = Self::set_icon_name, explicit_notify)]
pub icon_name: RefCell<String>,
pub extra_classes: RefCell<Vec<String>>,
#[property(get = Self::action_name, set = Self::set_action_name, override_interface = gtk::Actionable)]
pub action_name: RefCell<Option<glib::GString>>,
#[property(get = Self::action_target_value, set = Self::set_action_target, override_interface = gtk::Actionable)]
pub action_target: RefCell<Option<glib::Variant>>,
#[property(get, set = Self::set_state, explicit_notify, builder(ActionState::default()))]
pub state: Cell<ActionState>,
#[property(set = Self::set_default_state_tooltip_text)]
pub default_state_tooltip_text: PhantomData<Option<String>>,
#[template_child]
pub stack: TemplateChild<gtk::Stack>,
#[template_child]
pub button_default: TemplateChild<gtk::Button>,
#[template_child]
pub spinner: TemplateChild<Spinner>,
}
#[glib::object_subclass]
impl ObjectSubclass for ActionButton {
const NAME: &'static str = "ActionButton";
type Type = super::ActionButton;
type ParentType = adw::Bin;
type Interfaces = (gtk::Actionable,);
fn class_init(klass: &mut Self::Class) {
Self::bind_template(klass);
Self::Type::bind_template_callbacks(klass);
klass.set_css_name("action-button");
}
fn instance_init(obj: &InitializingObject<Self>) {
obj.init_template();
}
}
#[glib::derived_properties]
impl ObjectImpl for ActionButton {
fn signals() -> &'static [Signal] {
static SIGNALS: Lazy<Vec<Signal>> =
Lazy::new(|| vec![Signal::builder("clicked").build()]);
SIGNALS.as_ref()
}
}
impl WidgetImpl for ActionButton {}
impl BinImpl for ActionButton {}
impl ActionableImpl for ActionButton {
fn action_name(&self) -> Option<glib::GString> {
self.action_name.borrow().clone()
}
fn action_target_value(&self) -> Option<glib::Variant> {
self.action_target.borrow().clone()
}
fn set_action_name(&self, name: Option<&str>) {
self.action_name.replace(name.map(Into::into));
}
fn set_action_target_value(&self, value: Option<&glib::Variant>) {
self.set_action_target(value.cloned());
}
}
impl ActionButton {
fn set_icon_name(&self, icon_name: &str) {
if self.icon_name.borrow().as_str() == icon_name {
return;
}
self.icon_name.replace(icon_name.to_owned());
self.obj().notify_icon_name();
}
fn set_state(&self, state: ActionState) {
if self.state.get() == state {
return;
}
self.stack.set_visible_child_name(state.as_ref());
self.state.replace(state);
self.obj().notify_state();
}
fn set_action_target(&self, value: Option<glib::Variant>) {
self.action_target.replace(value);
}
fn set_default_state_tooltip_text(&self, text: Option<String>) {
self.button_default.set_tooltip_text(text.as_deref());
}
}
}
glib::wrapper! {
pub struct ActionButton(ObjectSubclass<imp::ActionButton>)
@extends gtk::Widget, adw::Bin, @implements gtk::Actionable, gtk::Accessible;
}
#[gtk::template_callbacks]
impl ActionButton {
pub fn new() -> Self {
glib::Object::new()
}
pub fn extra_classes(&self) -> Vec<String> {
self.imp().extra_classes.borrow().clone()
}
pub fn set_extra_classes(&self, classes: &[&str]) {
let imp = self.imp();
for class in imp.extra_classes.borrow_mut().drain(..) {
imp.button_default.remove_css_class(&class);
}
for class in classes.iter() {
imp.button_default.add_css_class(class);
}
self.imp()
.extra_classes
.replace(classes.iter().map(ToString::to_string).collect());
}
pub fn connect_clicked<F: Fn(&Self) + 'static>(&self, f: F) -> glib::SignalHandlerId {
self.connect_closure(
"clicked",
true,
closure_local!(move |obj: Self| {
f(&obj);
}),
)
}
#[template_callback]
fn button_clicked(&self) {
self.emit_by_name::<()>("clicked", &[]);
}
}