[fractal/wip/christopherdavis/use-hdyavatar: 2/2] avatar: Use HdyAvatar instead of a custom DrawingArea




commit 09b203a5629080f2fdab1b419dfaefdfdc587ba6
Author: Christopher Davis <brainblasted disroot org>
Date:   Mon Sep 28 23:23:09 2020 -0700

    avatar: Use HdyAvatar instead of a custom DrawingArea
    
    HdyAvatar allows for simple HiDPI image loading and
    a fallback image.
    
    Related to https://gitlab.gnome.org/GNOME/fractal/issues/496

 Cargo.lock                         |  50 ++-------------
 fractal-gtk/Cargo.toml             |   3 +-
 fractal-gtk/src/cache/mod.rs       |   6 +-
 fractal-gtk/src/widgets/avatar.rs  | 126 ++++++++++---------------------------
 fractal-gtk/src/widgets/message.rs |   1 +
 5 files changed, 43 insertions(+), 143 deletions(-)
---
diff --git a/Cargo.lock b/Cargo.lock
index 0873b493..f8a849da 100644
--- a/Cargo.lock
+++ b/Cargo.lock
@@ -567,7 +567,6 @@ dependencies = [
  "html2pango",
  "itertools 0.8.2",
  "lazy_static",
- "letter-avatar",
  "libhandy",
  "log",
  "loggerv",
@@ -1422,18 +1421,6 @@ version = "1.4.0"
 source = "registry+https://github.com/rust-lang/crates.io-index";
 checksum = "e2abad23fbc42b3700f2f279844dc832adb2b2eb069b2df918f455c4e18cc646"
 
-[[package]]
-name = "letter-avatar"
-version = "1.3.0"
-source = "registry+https://github.com/rust-lang/crates.io-index";
-checksum = "662660bd131a9c06b36730750de8049efb5d2c8e0c23b3b58e9dd87e61acc245"
-dependencies = [
- "cairo-rs",
- "pango",
- "pangocairo",
- "unicode-segmentation",
-]
-
 [[package]]
 name = "libc"
 version = "0.2.76"
@@ -1442,12 +1429,14 @@ checksum = "755456fae044e6fa1ebbbd1b3e902ae19e73097ed4ed87bb79934a867c007bc3"
 
 [[package]]
 name = "libhandy"
-version = "0.7.0"
+version = "0.7.1"
 source = "registry+https://github.com/rust-lang/crates.io-index";
-checksum = "ce64d9c1f9e9444fb1175cf65cb6db9b55e7ae5f9a44140202da05784a7cdf33"
+checksum = "d776bf5b92993b8006688652cda2683261317dcc82bb4d91d3996bda2f7019e0"
 dependencies = [
  "bitflags",
  "gdk",
+ "gdk-pixbuf",
+ "gdk-pixbuf-sys",
  "gdk-sys",
  "gio",
  "gio-sys",
@@ -1882,37 +1871,6 @@ dependencies = [
  "system-deps",
 ]
 
-[[package]]
-name = "pangocairo"
-version = "0.10.0"
-source = "registry+https://github.com/rust-lang/crates.io-index";
-checksum = "00f5ae67a05a5e023f09f64e9a71c845274d4b82dedee237b70425811885e883"
-dependencies = [
- "bitflags",
- "cairo-rs",
- "cairo-sys-rs",
- "glib",
- "glib-sys",
- "gobject-sys",
- "libc",
- "pango",
- "pango-sys",
- "pangocairo-sys",
-]
-
-[[package]]
-name = "pangocairo-sys"
-version = "0.11.0"
-source = "registry+https://github.com/rust-lang/crates.io-index";
-checksum = "94ccc97f698c2f0233b84e5ca676893a1e676785b60eec700b9c0e6dcd0feb98"
-dependencies = [
- "cairo-sys-rs",
- "glib-sys",
- "libc",
- "pango-sys",
- "system-deps",
-]
-
 [[package]]
 name = "paste"
 version = "0.1.18"
diff --git a/fractal-gtk/Cargo.toml b/fractal-gtk/Cargo.toml
index 04b8b794..8ab99f2a 100644
--- a/fractal-gtk/Cargo.toml
+++ b/fractal-gtk/Cargo.toml
@@ -28,7 +28,6 @@ rand = "0.7.3"
 regex = "1.3.4"
 secret-service = "1.0.0"
 serde_json = "1.0.48"
-letter-avatar = "1.3.0"
 sourceview4 = "0.2.0"
 gspell = "0.5.0"
 gtk-sys = "0.10.0"
@@ -54,7 +53,7 @@ path = "../fractal-matrix-api"
 package = "fractal-matrix-api"
 
 [dependencies.libhandy]
-version = "0.7.0"
+version = "0.7.1"
 
 [dependencies.gettext-rs]
 git = "https://github.com/danigm/gettext-rs";
diff --git a/fractal-gtk/src/cache/mod.rs b/fractal-gtk/src/cache/mod.rs
index 8943eac6..8a8dc78a 100644
--- a/fractal-gtk/src/cache/mod.rs
+++ b/fractal-gtk/src/cache/mod.rs
@@ -178,7 +178,7 @@ pub fn download_to_cache(
         Err(TryRecvError::Empty) => Continue(true),
         Err(TryRecvError::Disconnected) => Continue(false),
         Ok(_resp) => {
-            data.borrow_mut().redraw_pixbuf();
+            data.borrow_mut().redraw(None);
             Continue(false)
         }
     });
@@ -204,7 +204,7 @@ pub fn download_to_cache_username(
             label.set_text(&username);
             if let Some(ref rc_data) = avatar {
                 let mut data = rc_data.borrow_mut();
-                data.redraw_fallback(Some(username));
+                data.redraw(Some(username));
             }
 
             Continue(false)
@@ -235,7 +235,7 @@ pub fn download_to_cache_username_emote(
             label.set_markup(&format!("<b>{}</b> {}", &username, text));
             if let Some(ref rc_data) = avatar {
                 let mut data = rc_data.borrow_mut();
-                data.redraw_fallback(Some(username));
+                data.redraw(Some(username));
             }
 
             Continue(false)
diff --git a/fractal-gtk/src/widgets/avatar.rs b/fractal-gtk/src/widgets/avatar.rs
index f2a2a725..449d2725 100644
--- a/fractal-gtk/src/widgets/avatar.rs
+++ b/fractal-gtk/src/widgets/avatar.rs
@@ -1,12 +1,12 @@
+use std::boxed::Box;
 use std::cell::RefCell;
 use std::path::Path;
 use std::rc::Rc;
 
 use crate::util::cache_dir_path;
-use gdk::prelude::GdkContextExt;
 use gdk_pixbuf::Pixbuf;
 use gtk::prelude::*;
-pub use gtk::DrawingArea;
+use libhandy::AvatarExt as HdyAvatarExt;
 
 pub enum AvatarBadgeColor {
     Gold,
@@ -18,33 +18,27 @@ pub type Avatar = gtk::Overlay;
 
 pub struct AvatarData {
     id: String,
-    username: Option<String>,
-    size: i32,
-    cache: Option<Pixbuf>,
-    pub widget: gtk::DrawingArea,
-    fallback: cairo::ImageSurface,
+    pub widget: libhandy::Avatar,
 }
 
 impl AvatarData {
-    pub fn redraw_fallback(&mut self, username: Option<String>) {
-        self.username = username.clone();
-        /* This function should never fail */
-        self.fallback = letter_avatar::generate::new(self.id.clone(), username, self.size as f64)
-            .expect("this function should never fail");
-        self.widget.queue_draw();
-    }
-
-    pub fn redraw_pixbuf(&mut self) {
-        let path = cache_dir_path(None, self.id.as_str()).unwrap_or_default();
-        self.cache = load_pixbuf(&path, self.size);
-        self.widget.queue_draw();
+    pub fn redraw(&mut self, username: Option<String>) {
+        let id = self.id.clone();
+        if let Some(n) = username {
+            self.widget.set_text(Some(&n));
+        }
+        // Ensure that we reload the avatar
+        self.widget.set_image_load_func(Some(Box::new(move |sz| {
+            let path = cache_dir_path(None, &id).unwrap_or_default();
+            load_pixbuf(&path, sz)
+        })));
     }
 }
 
 pub trait AvatarExt {
     fn avatar_new(size: Option<i32>) -> gtk::Overlay;
     fn clean(&self);
-    fn create_da(&self, size: Option<i32>) -> DrawingArea;
+    fn create_avatar(&self, size: Option<i32>) -> libhandy::Avatar;
     fn circle(
         &self,
         id: String,
@@ -62,20 +56,19 @@ impl AvatarExt for gtk::Overlay {
         }
     }
 
-    fn create_da(&self, size: Option<i32>) -> DrawingArea {
-        let da = DrawingArea::new();
-
+    fn create_avatar(&self, size: Option<i32>) -> libhandy::Avatar {
         let s = size.unwrap_or(40);
-        da.set_size_request(s, s);
-        self.add(&da);
+        let avatar = libhandy::Avatar::new(s, None, true);
+        avatar.set_show_initials(true);
+        self.add(&avatar);
         self.show_all();
 
-        da
+        avatar
     }
 
     fn avatar_new(size: Option<i32>) -> gtk::Overlay {
         let b = gtk::Overlay::new();
-        b.create_da(size);
+        b.create_avatar(size);
         b.show_all();
         b.get_style_context().add_class("avatar");
 
@@ -96,21 +89,19 @@ impl AvatarExt for gtk::Overlay {
         badge_size: Option<i32>,
     ) -> Rc<RefCell<AvatarData>> {
         self.clean();
-        let da = self.create_da(Some(size));
-        let path = cache_dir_path(None, id.as_str()).unwrap_or_default();
-        let user_avatar = load_pixbuf(&path, size);
-        let uname = username.clone();
+        let avatar = self.create_avatar(Some(size));
         /* remove IRC postfix from the username */
         let username = if let Some(u) = username {
-            Some(u.trim_end_matches(" (IRC)").to_owned())
+            u.trim_end_matches(" (IRC)")
+                .trim_start_matches("#")
+                .to_owned()
         } else {
-            None
+            id.clone()
         };
-        /* This function should never fail */
-        let fallback = letter_avatar::generate::new(id.clone(), username, size as f64)
-            .expect("this function should never fail");
+
+        avatar.set_text(Some(&username));
+
         // Power level badge setup
-        let has_badge = badge_color.is_some();
         let badge_size = badge_size.unwrap_or(size / 3);
         if let Some(color) = badge_color {
             let badge = gtk::Box::new(gtk::Orientation::Vertical, 0);
@@ -127,64 +118,15 @@ impl AvatarExt for gtk::Overlay {
         }
 
         let data = AvatarData {
-            id,
-            username: uname,
-            size,
-            cache: user_avatar,
-            fallback,
-            widget: da.clone(),
+            id: id.clone(),
+            widget: avatar.clone(),
         };
         let avatar_cache: Rc<RefCell<AvatarData>> = Rc::new(RefCell::new(data));
 
-        let user_cache = avatar_cache.clone();
-        da.connect_draw(move |da, g| {
-            use std::f64::consts::PI;
-            let width = size as f64;
-            let height = size as f64;
-
-            g.set_antialias(cairo::Antialias::Best);
-
-            {
-                g.set_fill_rule(cairo::FillRule::EvenOdd);
-                g.arc(
-                    width / 2.0,
-                    height / 2.0,
-                    width.min(height) / 2.0,
-                    0.0,
-                    2.0 * PI,
-                );
-                if has_badge {
-                    g.clip_preserve();
-                    g.new_sub_path();
-                    let badge_radius = badge_size as f64 / 2.0;
-                    g.arc(
-                        width - badge_radius,
-                        badge_radius,
-                        badge_radius * 1.4,
-                        0.0,
-                        2.0 * PI,
-                    );
-                }
-                g.clip();
-
-                let data = user_cache.borrow();
-                if let Some(ref pb) = data.cache {
-                    let context = da.get_style_context();
-                    gtk::render_background(&context, g, 0.0, 0.0, width, height);
-
-                    let hpos: f64 = (width - (pb.get_height()) as f64) / 2.0;
-                    g.set_source_pixbuf(&pb, 0.0, hpos);
-                } else {
-                    /* use fallback */
-                    g.set_source_surface(&data.fallback, 0f64, 0f64);
-                }
-            }
-
-            g.rectangle(0.0, 0.0, width, height);
-            g.fill();
-
-            Inhibit(false)
-        });
+        avatar.set_image_load_func(Some(Box::new(move |sz| {
+            let path = cache_dir_path(None, &id).unwrap_or_default();
+            load_pixbuf(&path, sz)
+        })));
 
         avatar_cache
     }
diff --git a/fractal-gtk/src/widgets/message.rs b/fractal-gtk/src/widgets/message.rs
index 084e16e9..d68e98ec 100644
--- a/fractal-gtk/src/widgets/message.rs
+++ b/fractal-gtk/src/widgets/message.rs
@@ -268,6 +268,7 @@ impl MessageBox {
         let uid = msg.sender.clone();
         let alias = msg.sender_name.clone();
         let avatar = widgets::Avatar::avatar_new(Some(globals::MSG_ICON_SIZE));
+        avatar.set_valign(gtk::Align::Start);
 
         let data = avatar.circle(
             uid.to_string(),


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]