libcosmic/examples/dock/dock_item/mod.rs
2022-01-04 15:46:43 -05:00

146 lines
4.3 KiB
Rust

use cascade::cascade;
use gio::DesktopAppInfo;
use gio::Icon;
use gtk4::glib;
use gtk4::prelude::*;
use gtk4::subclass::prelude::*;
use gtk4::Align;
use gtk4::Box;
use gtk4::Image;
use gtk4::Label;
use gtk4::Orientation;
use gtk4::Popover;
use crate::dock_object::DockObject;
use crate::dock_popover::DockPopover;
use crate::utils::BoxedWindowList;
mod imp;
glib::wrapper! {
pub struct DockItem(ObjectSubclass<imp::DockItem>)
@extends gtk4::Button, gtk4::Widget,
@implements gtk4::Accessible, gtk4::Actionable, gtk4::Buildable, gtk4::ConstraintTarget;
}
impl Default for DockItem {
fn default() -> Self {
Self::new()
}
}
impl DockItem {
pub fn new() -> Self {
let self_: DockItem = glib::Object::new(&[]).expect("Failed to create DockItem");
let item_box = Box::new(Orientation::Vertical, 0);
self_.set_child(Some(&item_box));
self_.add_css_class("dock_item");
let image = cascade! {
Image::new();
..set_hexpand(true);
..set_halign(Align::Center);
..set_pixel_size(64);
};
let dots = cascade! {
Label::new(Some(""));
..set_hexpand(true);
..set_halign(Align::Center);
};
item_box.append(&image);
item_box.append(&dots);
let popover = cascade! {
Popover::new();
..set_autohide(true);
};
item_box.append(&popover);
let self_clone = self_.clone();
popover.connect_closed(move |_| {
self_clone
.emit_by_name::<&str>("popover-closed", &[])
.unwrap();
});
let popover_menu = cascade! {
DockPopover::new();
};
popover.set_child(Some(&popover_menu));
popover_menu
.connect_local(
"menu-hide",
false,
glib::clone!(@weak popover => @default-return None, move |_| {
popover.popdown();
None
}),
)
.unwrap();
let imp = imp::DockItem::from_instance(&self_);
imp.image.replace(image);
imp.dots.replace(dots);
imp.item_box.replace(item_box);
imp.popover.replace(popover);
imp.popover_menu.replace(popover_menu);
self_
}
// refactor to emit event for removing the item?
pub fn set_app_info(&self, dock_object: &DockObject) {
let self_ = imp::DockItem::from_instance(self);
if let Ok(app_info_value) = dock_object.property("appinfo") {
if let Ok(Some(app_info)) = app_info_value.get::<Option<DesktopAppInfo>>() {
self_
.image
.borrow()
.set_tooltip_text(Some(&app_info.name()));
let icon = app_info.icon().unwrap_or(
Icon::for_string("image-missing").expect("Failed to set default icon"),
);
self_.image.borrow().set_from_gicon(&icon);
}
} else {
println!("initializing dock item failed...");
}
if let Ok(active_value) = dock_object.property("active") {
if let Ok(active) = active_value.get::<BoxedWindowList>() {
let dots = self_.dots.borrow();
dots.set_text("");
for _ in active.0 {
dots.set_text(format!("{}{}", dots.text(), " · ").as_str());
}
}
}
if let Ok(popover_value) = dock_object.property("popover") {
if let Ok(popover) = popover_value.get::<bool>() {
dbg!(popover);
dbg!(dock_object);
if popover {
self.add_popover(dock_object);
} else {
self.clear_popover();
}
}
}
}
pub fn add_popover(&self, item: &DockObject) {
let imp = imp::DockItem::from_instance(self);
let popover = imp.popover.borrow();
let popover_menu = imp.popover_menu.borrow();
popover_menu.set_dock_object(item, true);
popover.popup();
}
pub fn clear_popover(&self) {
let imp = imp::DockItem::from_instance(self);
let popover = imp.popover.borrow();
popover.popdown();
}
}