use cascade::cascade; use gdk4::pango::EllipsizeMode; use gio::DesktopAppInfo; use gtk4::subclass::prelude::*; use gtk4::{gio, glib}; use gtk4::{prelude::*, Label}; use gtk4::{Box, Button, Image, ListBox, Orientation, Window}; use crate::dock_object::DockObject; use crate::utils::BoxedWindowList; use crate::Event; use crate::TX; mod imp; glib::wrapper! { pub struct DockPopover(ObjectSubclass) @extends gtk4::Widget, gtk4::Box, @implements gtk4::Accessible, gtk4::Buildable, gtk4::ConstraintTarget, gtk4::Orientable; } impl Default for DockPopover { fn default() -> Self { Self::new() } } impl DockPopover { pub fn new() -> Self { let self_: DockPopover = glib::Object::new(&[]).expect("Failed to create DockList"); self_.layout(); //dnd behavior is different for each type, as well as the data in the model self_ } pub fn set_dock_object(&self, dock_object: &DockObject, update_layout: bool) { let imp = imp::DockPopover::from_instance(&self); imp.dock_object.replace(Some(dock_object.clone())); if update_layout { self.update_layout(); } } pub fn update_layout(&self) { self.reset_menu(); cascade! { &self; ..set_spacing(4); ..set_orientation(Orientation::Vertical); ..set_hexpand(true); }; // build menu let imp = imp::DockPopover::from_instance(&self); let dock_object = imp.dock_object.borrow(); let menu_handle = imp.menu_handle.borrow(); if let Some(dock_object) = dock_object.as_ref() { if let Some(menu) = dock_object.get_popover_menu() { menu_handle.append(&menu); } else { let all_windows_item_container = cascade! { Box::new(Orientation::Vertical, 4); }; menu_handle.append(&all_windows_item_container); let window_list = dock_object.property::("active"); if window_list.0.len() == 0 { all_windows_item_container.hide(); } else { let window_listbox = cascade! { ListBox::new(); ..set_activate_on_single_click(true); }; all_windows_item_container.append(&window_listbox); for w in window_list.0 { let window_box = cascade! { Box::new(Orientation::Vertical, 4); }; window_listbox.append(&window_box); let window_title = cascade! { Label::new(Some(w.name.as_str())); ..set_margin_start(4); ..set_margin_end(4); ..set_margin_top(4); ..set_margin_bottom(4); ..set_wrap(true); ..set_max_width_chars(20); ..set_ellipsize(EllipsizeMode::End); ..add_css_class("title-4"); ..add_css_class("window_title"); }; let window_image = cascade! { //TODO fill with image of window Image::from_pixbuf(None); }; window_box.append(&window_image); window_box.append(&window_title); } // imp.all_windows_item_revealer.replace(window_list_revealer); imp.window_list.replace(window_listbox); } let launch_item_container = cascade! { Box::new(Orientation::Vertical, 4); ..set_hexpand(true); }; menu_handle.append(&launch_item_container); let launch_new_item = cascade! { Button::with_label("New Window"); }; launch_item_container.append(&launch_new_item); imp.launch_new_item.replace(launch_new_item); let favorite_item = cascade! { Button::with_label(if dock_object.property::("saved") {"Remove from Favorites"} else {"Add to Favorites"}); }; menu_handle.append(&favorite_item); imp.favorite_item.replace(favorite_item); let window_list = dock_object.property::("active"); if window_list.0.len() > 1 { let quit_all_item = cascade! { Button::with_label(format!("Quit {} Windows", window_list.0.len()).as_str()); }; menu_handle.append(&quit_all_item); imp.quit_all_item.replace(quit_all_item); } else { let quit_all_item = cascade! { Button::with_label("Quit"); }; menu_handle.append(&quit_all_item); if window_list.0.len() == 0 { quit_all_item.hide(); } imp.quit_all_item.replace(quit_all_item); } self.setup_handlers(); } } } fn layout(&self) { let imp = imp::DockPopover::from_instance(&self); let menu_handle = cascade! { Box::new(Orientation::Vertical, 4); }; self.append(&menu_handle); imp.menu_handle.replace(menu_handle); } fn emit_hide(&self) { self.emit_by_name::<()>("menu-hide", &[]); } pub fn reset_menu(&self) { // reset menu let menu_handle = cascade! { Box::new(Orientation::Vertical, 4); }; self.append(&menu_handle); let imp = imp::DockPopover::from_instance(&self); let old_menu_handle = imp.menu_handle.replace(menu_handle); self.remove(&old_menu_handle); } fn setup_handlers(&self) { let imp = imp::DockPopover::from_instance(&self); let dock_object = imp.dock_object.borrow(); let launch_new_item = imp.launch_new_item.borrow(); let favorite_item = imp.favorite_item.borrow(); let quit_all_item = imp.quit_all_item.borrow(); let window_listbox = imp.window_list.borrow(); // let all_windows_header = imp.all_windows_item_header.borrow(); // let revealer = &imp.all_windows_item_revealer; if let Some(dock_object) = dock_object.as_ref() { // println!("setting up popover menu handlers"); let self_ = self.clone(); launch_new_item.connect_clicked(glib::clone!(@weak dock_object, => move |_| { let app_info = dock_object.property::>("appinfo").expect("Failed to convert value to DesktopAppInfo"); let window = self_.root().unwrap().downcast::().unwrap(); let context = window.display().app_launch_context(); if let Err(err) = app_info.launch(&[], Some(&context)) { gtk4::MessageDialog::builder() .text(&format!("Failed to start {}", app_info.name())) .secondary_text(&err.to_string()) .message_type(gtk4::MessageType::Error) .modal(true) .transient_for(&window) .build() .show(); } self_.emit_hide(); })); let self_ = self.clone(); quit_all_item.connect_clicked(glib::clone!(@weak dock_object => move |_| { let active = dock_object.property::("active").0; for w in active { let entity = w.entity.clone(); glib::MainContext::default().spawn_local(async move { if let Some(tx) = TX.get() { let _ = tx.send(Event::Close(entity)).await; } }); } self_.emit_hide(); })); let self_ = self.clone(); favorite_item.connect_clicked(glib::clone!(@weak dock_object => move |_| { let saved = dock_object.property::("saved"); glib::MainContext::default().spawn_local(async move { if let Some(tx) = TX.get() { if let Some(name) = dock_object.get_name() { let _ = tx.send(Event::Favorite((name.into(), !saved))).await; } } }); self_.emit_hide(); })); // all_windows_header.connect_clicked( // glib::clone!(@weak dock_object, @weak revealer => move |self_| { // // dbg!(dock_object); // let revealer = revealer.borrow(); // revealer.set_reveal_child(!revealer.reveals_child()) // }), // ); let self_ = self.clone(); window_listbox.connect_row_activated( glib::clone!(@weak dock_object => move |_, item| { let active = dock_object.property::("active").0; let entity = active[usize::try_from(item.index()).unwrap()].entity.clone(); glib::MainContext::default().spawn_local(async move { if let Some(tx) = TX.get() { let _ = tx.send(Event::Activate(entity)).await; } }); self_.emit_hide(); }), ); } } }