fractal/components/avatar/
mod.rsuse adw::subclass::prelude::*;
use gtk::{glib, glib::clone, prelude::*, CompositeTemplate};
mod crop_circle;
mod data;
mod editable;
mod image;
mod overlapping;
use self::image::AvatarPaintableSize;
pub use self::{
data::AvatarData,
editable::EditableAvatar,
image::{AvatarImage, AvatarUriSource},
overlapping::OverlappingAvatars,
};
use crate::{
components::AnimatedImagePaintable,
utils::{BoundObject, BoundObjectWeakRef, CountedRef},
};
mod imp {
use std::{cell::RefCell, marker::PhantomData};
use glib::subclass::InitializingObject;
use super::*;
#[derive(Debug, Default, CompositeTemplate, glib::Properties)]
#[template(resource = "/org/gnome/Fractal/ui/components/avatar/mod.ui")]
#[properties(wrapper_type = super::Avatar)]
pub struct Avatar {
#[template_child]
avatar: TemplateChild<adw::Avatar>,
#[property(get, set = Self::set_data, explicit_notify, nullable)]
data: BoundObject<AvatarData>,
#[property(get)]
image: BoundObjectWeakRef<AvatarImage>,
#[property(get = Self::size, set = Self::set_size, explicit_notify, builder().default_value(-1).minimum(-1))]
size: PhantomData<i32>,
paintable_ref: RefCell<Option<CountedRef>>,
paintable_animation_ref: RefCell<Option<CountedRef>>,
}
#[glib::object_subclass]
impl ObjectSubclass for Avatar {
const NAME: &'static str = "Avatar";
type Type = super::Avatar;
type ParentType = adw::Bin;
fn class_init(klass: &mut Self::Class) {
AvatarImage::ensure_type();
Self::bind_template(klass);
Self::bind_template_callbacks(klass);
klass.set_accessible_role(gtk::AccessibleRole::Img);
}
fn instance_init(obj: &InitializingObject<Self>) {
obj.init_template();
}
}
#[glib::derived_properties]
impl ObjectImpl for Avatar {}
impl WidgetImpl for Avatar {
fn map(&self) {
self.parent_map();
self.update_paintable();
}
fn unmap(&self) {
self.parent_unmap();
self.update_animated_paintable_state();
}
}
impl BinImpl for Avatar {}
impl AccessibleImpl for Avatar {
fn first_accessible_child(&self) -> Option<gtk::Accessible> {
None
}
}
#[gtk::template_callbacks]
impl Avatar {
fn size(&self) -> i32 {
self.avatar.size()
}
fn set_size(&self, size: i32) {
if self.size() == size {
return;
}
self.avatar.set_size(size);
self.update_paintable();
self.obj().notify_size();
}
fn set_data(&self, data: Option<AvatarData>) {
if self.data.obj() == data {
return;
}
self.data.disconnect_signals();
if let Some(data) = data {
let image_handler = data.connect_image_notify(clone!(
#[weak(rename_to = imp)]
self,
move |_| {
imp.update_image();
}
));
self.data.set(data, vec![image_handler]);
}
self.update_image();
self.obj().notify_data();
}
fn update_image(&self) {
let image = self.data.obj().and_then(|data| data.image());
if self.image.obj() == image {
return;
}
self.image.disconnect_signals();
if let Some(image) = &image {
let small_paintable_handler = image.connect_small_paintable_notify(clone!(
#[weak(rename_to = imp)]
self,
move |_| {
imp.update_paintable();
}
));
let big_paintable_handler = image.connect_big_paintable_notify(clone!(
#[weak(rename_to = imp)]
self,
move |_| {
imp.update_paintable();
}
));
self.image
.set(image, vec![small_paintable_handler, big_paintable_handler]);
}
self.update_scale_factor();
self.update_paintable();
self.obj().notify_image();
}
fn needs_small_paintable(&self) -> bool {
AvatarPaintableSize::from(self.size()) == AvatarPaintableSize::Small
}
#[template_callback]
fn update_scale_factor(&self) {
let Some(image) = self.image.obj() else {
return;
};
let scale_factor = self.obj().scale_factor().try_into().unwrap_or(1);
image.set_scale_factor(scale_factor);
}
fn update_paintable(&self) {
let _old_paintable_ref = self.paintable_ref.take();
if !self.obj().is_mapped() {
self.update_animated_paintable_state();
return;
}
let Some(image) = self.image.obj() else {
self.update_animated_paintable_state();
return;
};
let (paintable, paintable_ref) = if self.needs_small_paintable() {
(image.small_paintable(), image.small_paintable_ref())
} else {
(
image.big_paintable().or_else(|| image.small_paintable()),
image.big_paintable_ref(),
)
};
self.avatar.set_custom_image(paintable.as_ref());
self.paintable_ref.replace(Some(paintable_ref));
self.update_animated_paintable_state();
}
fn update_animated_paintable_state(&self) {
let _old_paintable_animation_ref = self.paintable_animation_ref.take();
if !self.obj().is_mapped() {
return;
}
let Some(image) = self.image.obj() else {
return;
};
let paintable = if self.needs_small_paintable() {
image.small_paintable()
} else {
image.big_paintable()
};
let Some(paintable) = paintable.and_downcast::<AnimatedImagePaintable>() else {
return;
};
self.paintable_animation_ref
.replace(Some(paintable.animation_ref()));
}
}
}
glib::wrapper! {
pub struct Avatar(ObjectSubclass<imp::Avatar>)
@extends gtk::Widget, adw::Bin, @implements gtk::Accessible;
}
impl Avatar {
pub fn new() -> Self {
glib::Object::new()
}
}