use std::time::Duration;
use adw::subclass::prelude::*;
use gettextrs::gettext;
use gtk::{
gdk, gio, glib,
glib::{clone, closure, closure_local},
prelude::*,
CompositeTemplate,
};
use tracing::{debug, error};
use super::{AvatarData, AvatarImage};
use crate::{
components::{ActionButton, ActionState, AnimatedImagePaintable},
toast,
utils::{
expression,
media::image::{load_image, ImageDimensions},
CountedRef,
},
};
#[derive(Debug, Default, Hash, Eq, PartialEq, Clone, Copy, glib::Enum)]
#[repr(u32)]
#[enum_type(name = "EditableAvatarState")]
pub enum EditableAvatarState {
#[default]
Default = 0,
EditInProgress = 1,
EditSuccessful = 2,
RemovalInProgress = 3,
}
mod imp {
use std::cell::{Cell, RefCell};
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/avatar/editable.ui")]
#[properties(wrapper_type = super::EditableAvatar)]
pub struct EditableAvatar {
#[property(get, set = Self::set_data, explicit_notify)]
pub data: RefCell<Option<AvatarData>>,
#[property(get, set = Self::set_editable, explicit_notify)]
pub editable: Cell<bool>,
#[property(get, set = Self::set_inhibit_remove, explicit_notify)]
pub inhibit_remove: Cell<bool>,
#[property(get, set = Self::set_state, explicit_notify, builder(EditableAvatarState::default()))]
pub state: Cell<EditableAvatarState>,
pub edit_state: Cell<ActionState>,
pub edit_sensitive: Cell<bool>,
pub removable: Cell<bool>,
pub remove_state: Cell<ActionState>,
pub remove_sensitive: Cell<bool>,
#[property(get)]
pub temp_paintable: RefCell<Option<gdk::Paintable>>,
temp_paintable_animation_ref: RefCell<Option<CountedRef>>,
#[template_child]
pub stack: TemplateChild<gtk::Stack>,
#[template_child]
pub temp_avatar: TemplateChild<adw::Avatar>,
#[template_child]
pub button_remove: TemplateChild<ActionButton>,
#[template_child]
pub button_edit: TemplateChild<ActionButton>,
}
#[glib::object_subclass]
impl ObjectSubclass for EditableAvatar {
const NAME: &'static str = "EditableAvatar";
type Type = super::EditableAvatar;
type ParentType = adw::Bin;
fn class_init(klass: &mut Self::Class) {
Self::bind_template(klass);
klass.set_css_name("editable-avatar");
klass.install_action_async(
"editable-avatar.edit-avatar",
None,
|obj, _, _| async move {
obj.choose_avatar().await;
},
);
klass.install_action("editable-avatar.remove-avatar", None, |obj, _, _| {
obj.emit_by_name::<()>("remove-avatar", &[]);
});
}
fn instance_init(obj: &InitializingObject<Self>) {
obj.init_template();
}
}
#[glib::derived_properties]
impl ObjectImpl for EditableAvatar {
fn signals() -> &'static [Signal] {
static SIGNALS: Lazy<Vec<Signal>> = Lazy::new(|| {
vec![
Signal::builder("edit-avatar")
.param_types([gio::File::static_type()])
.build(),
Signal::builder("remove-avatar").build(),
]
});
SIGNALS.as_ref()
}
fn constructed(&self) {
self.parent_constructed();
self.button_remove.set_extra_classes(&["error"]);
let obj = self.obj();
let image_present_expr = obj
.property_expression("data")
.chain_property::<AvatarData>("image")
.chain_property::<AvatarImage>("paintable")
.chain_closure::<bool>(closure!(
|_: Option<glib::Object>, image: Option<gdk::Paintable>| { image.is_some() }
));
let editable_expr = obj.property_expression("editable");
let remove_not_inhibited_expr =
expression::not(obj.property_expression("inhibit-remove"));
let can_remove_expr = expression::and(editable_expr, remove_not_inhibited_expr);
let button_remove_visible = expression::and(can_remove_expr, image_present_expr);
button_remove_visible.bind(&*self.button_remove, "visible", glib::Object::NONE);
self.temp_avatar.connect_map(clone!(
#[weak(rename_to = imp)]
self,
move |_| {
imp.update_temp_paintable_state();
}
));
self.temp_avatar.connect_unmap(clone!(
#[weak(rename_to = imp)]
self,
move |_| {
imp.update_temp_paintable_state();
}
));
}
}
impl WidgetImpl for EditableAvatar {}
impl BinImpl for EditableAvatar {}
impl EditableAvatar {
fn set_data(&self, data: Option<AvatarData>) {
if *self.data.borrow() == data {
return;
}
self.data.replace(data);
self.obj().notify_data();
}
fn set_editable(&self, editable: bool) {
if self.editable.get() == editable {
return;
}
self.editable.set(editable);
self.obj().notify_editable();
}
fn set_inhibit_remove(&self, inhibit: bool) {
if self.inhibit_remove.get() == inhibit {
return;
}
self.inhibit_remove.set(inhibit);
self.obj().notify_inhibit_remove();
}
fn set_state(&self, state: EditableAvatarState) {
if self.state.get() == state {
return;
}
let obj = self.obj();
match state {
EditableAvatarState::Default => {
self.show_temp_paintable(false);
obj.set_edit_state(ActionState::Default);
obj.set_edit_sensitive(true);
obj.set_remove_state(ActionState::Default);
obj.set_remove_sensitive(true);
self.set_temp_paintable(None);
}
EditableAvatarState::EditInProgress => {
self.show_temp_paintable(true);
obj.set_edit_state(ActionState::Loading);
obj.set_edit_sensitive(true);
obj.set_remove_state(ActionState::Default);
obj.set_remove_sensitive(false);
}
EditableAvatarState::EditSuccessful => {
self.show_temp_paintable(false);
obj.set_edit_sensitive(true);
obj.set_remove_state(ActionState::Default);
obj.set_remove_sensitive(true);
self.set_temp_paintable(None);
obj.set_edit_state(ActionState::Success);
glib::timeout_add_local_once(
Duration::from_secs(2),
clone!(
#[weak]
obj,
move || {
obj.set_state(EditableAvatarState::Default);
}
),
);
}
EditableAvatarState::RemovalInProgress => {
self.show_temp_paintable(true);
obj.set_edit_state(ActionState::Default);
obj.set_edit_sensitive(false);
obj.set_remove_state(ActionState::Loading);
obj.set_remove_sensitive(true);
}
}
self.state.set(state);
obj.notify_state();
}
fn avatar_dimensions(&self) -> ImageDimensions {
let scale_factor = self.obj().scale_factor();
let avatar_size = self.temp_avatar.size();
let size = (avatar_size * scale_factor) as u32;
ImageDimensions {
width: size,
height: size,
}
}
pub(super) async fn set_temp_paintable_from_file(&self, file: gio::File) {
let paintable = load_image(file, Some(self.avatar_dimensions())).await.ok();
self.set_temp_paintable(paintable);
}
fn set_temp_paintable(&self, paintable: Option<gdk::Paintable>) {
if *self.temp_paintable.borrow() == paintable {
return;
}
self.temp_paintable.replace(paintable);
self.update_temp_paintable_state();
self.obj().notify_temp_paintable();
}
fn show_temp_paintable(&self, show: bool) {
let stack = &self.stack;
if show {
stack.set_visible_child_name("temp");
} else {
stack.set_visible_child_name("default");
}
}
fn update_temp_paintable_state(&self) {
self.temp_paintable_animation_ref.take();
let Some(paintable) = self
.temp_paintable
.borrow()
.clone()
.and_downcast::<AnimatedImagePaintable>()
else {
return;
};
if self.temp_avatar.is_mapped() {
self.temp_paintable_animation_ref
.replace(Some(paintable.animation_ref()));
}
}
}
}
glib::wrapper! {
pub struct EditableAvatar(ObjectSubclass<imp::EditableAvatar>)
@extends gtk::Widget, adw::Bin, @implements gtk::Accessible;
}
impl EditableAvatar {
pub fn new() -> Self {
glib::Object::new()
}
pub fn reset(&self) {
self.set_state(EditableAvatarState::Default);
}
pub fn edit_in_progress(&self) {
self.set_state(EditableAvatarState::EditInProgress);
}
pub fn removal_in_progress(&self) {
self.set_state(EditableAvatarState::RemovalInProgress);
}
pub fn success(&self) {
if self.edit_state() == ActionState::Loading {
self.set_state(EditableAvatarState::EditSuccessful);
} else if self.remove_state() == ActionState::Loading {
self.set_state(EditableAvatarState::Default);
}
}
fn edit_state(&self) -> ActionState {
self.imp().edit_state.get()
}
fn set_edit_state(&self, state: ActionState) {
if self.edit_state() == state {
return;
}
self.imp().edit_state.set(state);
}
fn edit_sensitive(&self) -> bool {
self.imp().edit_sensitive.get()
}
fn set_edit_sensitive(&self, sensitive: bool) {
if self.edit_sensitive() == sensitive {
return;
}
self.imp().edit_sensitive.set(sensitive);
}
fn remove_state(&self) -> ActionState {
self.imp().remove_state.get()
}
fn set_remove_state(&self, state: ActionState) {
if self.remove_state() == state {
return;
}
self.imp().remove_state.set(state);
}
fn remove_sensitive(&self) -> bool {
self.imp().remove_sensitive.get()
}
fn set_remove_sensitive(&self, sensitive: bool) {
if self.remove_sensitive() == sensitive {
return;
}
self.imp().remove_sensitive.set(sensitive);
}
async fn choose_avatar(&self) {
let filters = gio::ListStore::new::<gtk::FileFilter>();
let image_filter = gtk::FileFilter::new();
image_filter.set_name(Some(&gettext("Images")));
image_filter.add_mime_type("image/*");
filters.append(&image_filter);
let dialog = gtk::FileDialog::builder()
.title(gettext("Choose Avatar"))
.modal(true)
.accept_label(gettext("Choose"))
.filters(&filters)
.build();
let file = match dialog
.open_future(self.root().and_downcast_ref::<gtk::Window>())
.await
{
Ok(file) => file,
Err(error) => {
if error.matches(gtk::DialogError::Dismissed) {
debug!("File dialog dismissed by user");
} else {
error!("Could not open avatar file: {error:?}");
toast!(self, gettext("Could not open avatar file"));
}
return;
}
};
if let Some(content_type) = file
.query_info_future(
gio::FILE_ATTRIBUTE_STANDARD_CONTENT_TYPE,
gio::FileQueryInfoFlags::NONE,
glib::Priority::LOW,
)
.await
.ok()
.and_then(|info| info.content_type())
{
if gio::content_type_is_a(&content_type, "image/*") {
self.imp().set_temp_paintable_from_file(file.clone()).await;
self.emit_by_name::<()>("edit-avatar", &[&file]);
} else {
error!("The chosen file is not an image");
toast!(self, gettext("The chosen file is not an image"));
}
} else {
error!("Could not get the content type of the file");
toast!(
self,
gettext("Could not determine the type of the chosen file")
);
}
}
pub fn connect_edit_avatar<F: Fn(&Self, gio::File) + 'static>(
&self,
f: F,
) -> glib::SignalHandlerId {
self.connect_closure(
"edit-avatar",
true,
closure_local!(|obj: Self, file: gio::File| {
f(&obj, file);
}),
)
}
pub fn connect_remove_avatar<F: Fn(&Self) + 'static>(&self, f: F) -> glib::SignalHandlerId {
self.connect_closure(
"remove-avatar",
true,
closure_local!(|obj: Self| {
f(&obj);
}),
)
}
}