#include "modules/privacy/privacy_item.hpp" #include #include #include #include #include #include #include #include "AModule.hpp" #include "glibmm/main.h" #include "glibmm/priorities.h" #include "gtkmm/enums.h" #include "gtkmm/label.h" #include "gtkmm/revealer.h" #include "gtkmm/tooltip.h" #include "sigc++/adaptors/bind.h" #include "util/gtk_icon.hpp" #include "util/pipewire/privacy_node_info.hpp" namespace waybar::modules::privacy { PrivacyItem::PrivacyItem(const Json::Value &config_, enum PrivacyNodeType privacy_type_, std::list *nodes_, const std::string &pos, const uint icon_size, const uint transition_duration) : Gtk::Revealer(), privacy_type(privacy_type_), nodes(nodes_), signal_conn(), tooltip_window(Gtk::ORIENTATION_VERTICAL, 0), box_(Gtk::ORIENTATION_HORIZONTAL, 0), icon_() { switch (privacy_type) { case util::PipewireBackend::PRIVACY_NODE_TYPE_AUDIO_INPUT: box_.get_style_context()->add_class("audio-in"); iconName = "waybar-privacy-audio-input-symbolic"; break; case util::PipewireBackend::PRIVACY_NODE_TYPE_AUDIO_OUTPUT: box_.get_style_context()->add_class("audio-out"); iconName = "waybar-privacy-audio-output-symbolic"; break; case util::PipewireBackend::PRIVACY_NODE_TYPE_VIDEO_INPUT: box_.get_style_context()->add_class("screenshare"); iconName = "waybar-privacy-screen-share-symbolic"; break; default: case util::PipewireBackend::PRIVACY_NODE_TYPE_NONE: return; } // Set the reveal transition to not look weird when sliding in if (pos == "modules-left") { set_transition_type(Gtk::REVEALER_TRANSITION_TYPE_SLIDE_RIGHT); } else if (pos == "modules-center") { set_transition_type(Gtk::REVEALER_TRANSITION_TYPE_CROSSFADE); } else if (pos == "modules-right") { set_transition_type(Gtk::REVEALER_TRANSITION_TYPE_SLIDE_LEFT); } set_transition_duration(transition_duration); box_.set_name("privacy-item"); box_.add(icon_); icon_.set_pixel_size(icon_size); add(box_); // Icon Name if (config_["icon-name"].isString()) { iconName = config_["icon-name"].asString(); } icon_.set_from_icon_name(iconName, Gtk::ICON_SIZE_INVALID); // Tooltip Icon Size if (config_["tooltip-icon-size"].isUInt()) { tooltipIconSize = config_["tooltip-icon-size"].asUInt(); } // Tooltip if (config_["tooltip"].isString()) { tooltip = config_["tooltip"].asBool(); } set_has_tooltip(tooltip); if (tooltip) { // Sets the window to use when showing the tooltip update_tooltip(); this->signal_query_tooltip().connect(sigc::track_obj( [this](int x, int y, bool keyboard_tooltip, const Glib::RefPtr &tooltip) { tooltip->set_custom(tooltip_window); return true; }, *this)); } // Don't show by default set_reveal_child(true); set_visible(false); } void PrivacyItem::update_tooltip() { // Removes all old nodes for (auto child : tooltip_window.get_children()) { delete child; } for (auto *node : *nodes) { Gtk::Box *box = new Gtk::Box(Gtk::ORIENTATION_HORIZONTAL, 4); // Set device icon Gtk::Image *node_icon = new Gtk::Image(); node_icon->set_pixel_size(tooltipIconSize); node_icon->set_from_icon_name(node->get_icon_name(), Gtk::ICON_SIZE_INVALID); box->add(*node_icon); // Set model Gtk::Label *node_name = new Gtk::Label(node->get_name()); box->add(*node_name); tooltip_window.add(*box); } tooltip_window.show_all(); } void PrivacyItem::set_in_use(bool in_use) { if (in_use) { update_tooltip(); } if (this->in_use == in_use && init) return; if (init) { // Disconnect any previous connection so that it doesn't get activated in // the future, hiding the module when it should be visible signal_conn.disconnect(); this->in_use = in_use; guint duration = 0; if (this->in_use) { set_visible(true); } else { set_reveal_child(false); duration = get_transition_duration(); } signal_conn = Glib::signal_timeout().connect(sigc::track_obj( [this] { if (this->in_use) { set_reveal_child(true); } else { set_visible(false); } return false; }, *this), duration); } else { set_visible(false); set_reveal_child(false); } this->init = true; } } // namespace waybar::modules::privacy