mirror of
https://github.com/Zedfrigg/ironbar.git
synced 2025-07-01 10:41:03 +02:00
feat: wlroots-agnostic support for launcher
module
This commit is contained in:
parent
bb4fe7f7f5
commit
b1c66b9117
7 changed files with 172 additions and 247 deletions
|
@ -13,7 +13,6 @@ mod wayland;
|
||||||
use crate::bar::create_bar;
|
use crate::bar::create_bar;
|
||||||
use crate::config::{Config, MonitorConfig};
|
use crate::config::{Config, MonitorConfig};
|
||||||
use crate::style::load_css;
|
use crate::style::load_css;
|
||||||
use crate::sway::get_client;
|
|
||||||
use color_eyre::eyre::Result;
|
use color_eyre::eyre::Result;
|
||||||
use color_eyre::Report;
|
use color_eyre::Report;
|
||||||
use dirs::config_dir;
|
use dirs::config_dir;
|
||||||
|
|
|
@ -47,10 +47,10 @@ impl Module<gtk::Box> for FocusedModule {
|
||||||
.expect("Failed to get read lock on toplevels")
|
.expect("Failed to get read lock on toplevels")
|
||||||
.clone();
|
.clone();
|
||||||
|
|
||||||
toplevels.into_iter().find(|top| top.active)
|
toplevels.into_iter().find(|(top, _)| top.active)
|
||||||
});
|
});
|
||||||
|
|
||||||
if let Some(top) = focused {
|
if let Some((top, _)) = focused {
|
||||||
tx.try_send(ModuleUpdateEvent::Update((
|
tx.try_send(ModuleUpdateEvent::Update((
|
||||||
top.title.clone(),
|
top.title.clone(),
|
||||||
top.app_id
|
top.app_id
|
||||||
|
|
|
@ -4,22 +4,20 @@ use crate::icon::get_icon;
|
||||||
use crate::modules::launcher::{ItemEvent, LauncherUpdate};
|
use crate::modules::launcher::{ItemEvent, LauncherUpdate};
|
||||||
use crate::modules::ModuleUpdateEvent;
|
use crate::modules::ModuleUpdateEvent;
|
||||||
use crate::popup::Popup;
|
use crate::popup::Popup;
|
||||||
use crate::sway::node::{get_node_id, is_node_xwayland};
|
|
||||||
use gtk::prelude::*;
|
use gtk::prelude::*;
|
||||||
use gtk::{Button, IconTheme, Image};
|
use gtk::{Button, IconTheme, Image};
|
||||||
use std::rc::Rc;
|
use std::rc::Rc;
|
||||||
use std::sync::RwLock;
|
use std::sync::RwLock;
|
||||||
use swayipc_async::Node;
|
|
||||||
use tokio::sync::mpsc::Sender;
|
use tokio::sync::mpsc::Sender;
|
||||||
|
use crate::wayland::ToplevelInfo;
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
pub struct Item {
|
pub struct Item {
|
||||||
pub app_id: String,
|
pub app_id: String,
|
||||||
pub favorite: bool,
|
pub favorite: bool,
|
||||||
pub open_state: OpenState,
|
pub open_state: OpenState,
|
||||||
pub windows: Collection<i64, Window>,
|
pub windows: Collection<usize, Window>,
|
||||||
pub name: Option<String>,
|
pub name: String,
|
||||||
pub is_xwayland: bool,
|
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Item {
|
impl Item {
|
||||||
|
@ -29,21 +27,18 @@ impl Item {
|
||||||
favorite,
|
favorite,
|
||||||
open_state,
|
open_state,
|
||||||
windows: Collection::new(),
|
windows: Collection::new(),
|
||||||
name: None,
|
name: String::new(),
|
||||||
is_xwayland: false,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Merges the provided node into this launcher item
|
/// Merges the provided node into this launcher item
|
||||||
pub fn merge_node(&mut self, node: Node) -> Window {
|
pub fn merge_toplevel(&mut self, node: ToplevelInfo) -> Window {
|
||||||
let id = node.id;
|
let id = node.id;
|
||||||
|
|
||||||
if self.windows.is_empty() {
|
if self.windows.is_empty() {
|
||||||
self.name = node.name.clone();
|
self.name = node.title.clone();
|
||||||
}
|
}
|
||||||
|
|
||||||
self.is_xwayland = self.is_xwayland || is_node_xwayland(&node);
|
|
||||||
|
|
||||||
let window: Window = node.into();
|
let window: Window = node.into();
|
||||||
self.windows.insert(id, window.clone());
|
self.windows.insert(id, window.clone());
|
||||||
|
|
||||||
|
@ -52,16 +47,12 @@ impl Item {
|
||||||
window
|
window
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn unmerge_node(&mut self, node: &Node) {
|
pub fn unmerge_toplevel(&mut self, node: &ToplevelInfo) {
|
||||||
self.windows.remove(&node.id);
|
self.windows.remove(&node.id);
|
||||||
self.recalculate_open_state();
|
self.recalculate_open_state();
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn get_name(&self) -> &str {
|
pub fn set_window_name(&mut self, window_id: usize, name: String) {
|
||||||
self.name.as_ref().unwrap_or(&self.app_id)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn set_window_name(&mut self, window_id: i64, name: Option<String>) {
|
|
||||||
if let Some(window) = self.windows.get_mut(&window_id) {
|
if let Some(window) = self.windows.get_mut(&window_id) {
|
||||||
if let OpenState::Open { focused: true, .. } = window.open_state {
|
if let OpenState::Open { focused: true, .. } = window.open_state {
|
||||||
self.name = name.clone();
|
self.name = name.clone();
|
||||||
|
@ -71,23 +62,7 @@ impl Item {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_unfocused(&mut self) {
|
pub fn set_window_focused(&mut self, window_id: usize, focused: bool) {
|
||||||
let focused = self
|
|
||||||
.windows
|
|
||||||
.iter_mut()
|
|
||||||
.find(|window| window.open_state.is_focused());
|
|
||||||
|
|
||||||
if let Some(focused) = focused {
|
|
||||||
focused.open_state = OpenState::Open {
|
|
||||||
focused: false,
|
|
||||||
urgent: focused.open_state.is_urgent(),
|
|
||||||
};
|
|
||||||
|
|
||||||
self.recalculate_open_state();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn set_window_focused(&mut self, window_id: i64, focused: bool) {
|
|
||||||
if let Some(window) = self.windows.get_mut(&window_id) {
|
if let Some(window) = self.windows.get_mut(&window_id) {
|
||||||
window.open_state =
|
window.open_state =
|
||||||
OpenState::merge_states(&[&window.open_state, &OpenState::focused(focused)]);
|
OpenState::merge_states(&[&window.open_state, &OpenState::focused(focused)]);
|
||||||
|
@ -96,15 +71,6 @@ impl Item {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_window_urgent(&mut self, window_id: i64, urgent: bool) {
|
|
||||||
if let Some(window) = self.windows.get_mut(&window_id) {
|
|
||||||
window.open_state =
|
|
||||||
OpenState::merge_states(&[&window.open_state, &OpenState::urgent(urgent)]);
|
|
||||||
|
|
||||||
self.recalculate_open_state();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Sets this item's open state
|
/// Sets this item's open state
|
||||||
/// to the merged result of its windows' open states
|
/// to the merged result of its windows' open states
|
||||||
fn recalculate_open_state(&mut self) {
|
fn recalculate_open_state(&mut self) {
|
||||||
|
@ -119,16 +85,14 @@ impl Item {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<Node> for Item {
|
impl From<ToplevelInfo> for Item {
|
||||||
fn from(node: Node) -> Self {
|
fn from(toplevel: ToplevelInfo) -> Self {
|
||||||
let app_id = get_node_id(&node).to_string();
|
let open_state = OpenState::from_toplevel(&toplevel);
|
||||||
let open_state = OpenState::from_node(&node);
|
let name = toplevel.title.clone();
|
||||||
let name = node.name.clone();
|
let app_id = toplevel.app_id.clone();
|
||||||
|
|
||||||
let is_xwayland = is_node_xwayland(&node);
|
|
||||||
|
|
||||||
let mut windows = Collection::new();
|
let mut windows = Collection::new();
|
||||||
windows.insert(node.id, node.into());
|
windows.insert(toplevel.id, toplevel.into());
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
app_id,
|
app_id,
|
||||||
|
@ -136,25 +100,24 @@ impl From<Node> for Item {
|
||||||
open_state,
|
open_state,
|
||||||
windows,
|
windows,
|
||||||
name,
|
name,
|
||||||
is_xwayland,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone, Debug)]
|
#[derive(Clone, Debug)]
|
||||||
pub struct Window {
|
pub struct Window {
|
||||||
pub id: i64,
|
pub id: usize,
|
||||||
pub name: Option<String>,
|
pub name: String,
|
||||||
pub open_state: OpenState,
|
pub open_state: OpenState,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl From<Node> for Window {
|
impl From<ToplevelInfo> for Window {
|
||||||
fn from(node: Node) -> Self {
|
fn from(node: ToplevelInfo) -> Self {
|
||||||
let open_state = OpenState::from_node(&node);
|
let open_state = OpenState::from_toplevel(&node);
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
id: node.id,
|
id: node.id,
|
||||||
name: node.name,
|
name: node.title,
|
||||||
open_state,
|
open_state,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -183,7 +146,7 @@ impl ItemButton {
|
||||||
let mut button = Button::builder();
|
let mut button = Button::builder();
|
||||||
|
|
||||||
if show_names {
|
if show_names {
|
||||||
button = button.label(item.get_name());
|
button = button.label(&item.name);
|
||||||
}
|
}
|
||||||
|
|
||||||
if show_icons {
|
if show_icons {
|
||||||
|
@ -208,9 +171,6 @@ impl ItemButton {
|
||||||
if item.open_state.is_focused() {
|
if item.open_state.is_focused() {
|
||||||
style_context.add_class("focused");
|
style_context.add_class("focused");
|
||||||
}
|
}
|
||||||
if item.open_state.is_urgent() {
|
|
||||||
style_context.add_class("urgent");
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
{
|
||||||
let app_id = item.app_id.clone();
|
let app_id = item.app_id.clone();
|
||||||
|
@ -274,7 +234,6 @@ impl ItemButton {
|
||||||
|
|
||||||
if !open {
|
if !open {
|
||||||
self.set_focused(false);
|
self.set_focused(false);
|
||||||
self.set_urgent(false);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -282,10 +241,6 @@ impl ItemButton {
|
||||||
self.update_class("focused", focused);
|
self.update_class("focused", focused);
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn set_urgent(&self, urgent: bool) {
|
|
||||||
self.update_class("urgent", urgent);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Adds or removes a class to the button based on `toggle`.
|
/// Adds or removes a class to the button based on `toggle`.
|
||||||
fn update_class(&self, class: &str, toggle: bool) {
|
fn update_class(&self, class: &str, toggle: bool) {
|
||||||
let style_context = self.button.style_context();
|
let style_context = self.button.style_context();
|
||||||
|
|
|
@ -1,14 +1,13 @@
|
||||||
mod item;
|
mod item;
|
||||||
mod open_state;
|
mod open_state;
|
||||||
|
|
||||||
|
use self::item::{Item, ItemButton, Window};
|
||||||
|
use self::open_state::OpenState;
|
||||||
use crate::collection::Collection;
|
use crate::collection::Collection;
|
||||||
use crate::icon::find_desktop_file;
|
use crate::icon::find_desktop_file;
|
||||||
use crate::modules::launcher::item::{Item, ItemButton, Window};
|
|
||||||
use crate::modules::launcher::open_state::OpenState;
|
|
||||||
use crate::modules::{Module, ModuleInfo, ModuleUpdateEvent, ModuleWidget, WidgetContext};
|
use crate::modules::{Module, ModuleInfo, ModuleUpdateEvent, ModuleWidget, WidgetContext};
|
||||||
use crate::sway::get_sub_client;
|
use crate::wayland;
|
||||||
use crate::sway::node::{get_node_id, get_open_windows};
|
use crate::wayland::ToplevelChange;
|
||||||
use crate::{await_sync, get_client};
|
|
||||||
use color_eyre::{Help, Report};
|
use color_eyre::{Help, Report};
|
||||||
use glib::Continue;
|
use glib::Continue;
|
||||||
use gtk::prelude::*;
|
use gtk::prelude::*;
|
||||||
|
@ -16,7 +15,6 @@ use gtk::{Button, IconTheme, Orientation};
|
||||||
use serde::Deserialize;
|
use serde::Deserialize;
|
||||||
use std::process::{Command, Stdio};
|
use std::process::{Command, Stdio};
|
||||||
use std::sync::{Arc, Mutex};
|
use std::sync::{Arc, Mutex};
|
||||||
use swayipc_async::WindowChange;
|
|
||||||
use tokio::spawn;
|
use tokio::spawn;
|
||||||
use tokio::sync::mpsc;
|
use tokio::sync::mpsc;
|
||||||
use tokio::sync::mpsc::{Receiver, Sender};
|
use tokio::sync::mpsc::{Receiver, Sender};
|
||||||
|
@ -47,13 +45,11 @@ pub enum LauncherUpdate {
|
||||||
/// Removes item with `app_id`
|
/// Removes item with `app_id`
|
||||||
RemoveItem(String),
|
RemoveItem(String),
|
||||||
/// Removes window from item with `app_id`.
|
/// Removes window from item with `app_id`.
|
||||||
RemoveWindow(String, i64),
|
RemoveWindow(String, usize),
|
||||||
/// Sets title for `app_id`
|
/// Sets title for `app_id`
|
||||||
Title(String, i64, Option<String>),
|
Title(String, usize, String),
|
||||||
/// Focuses first `app_id`, unfocuses second `app_id` (if present)
|
/// Marks the item with `app_id` as focused or not focused
|
||||||
Focus(String, Option<String>),
|
Focus(String, bool),
|
||||||
/// Marks the item with `app_id` as urgent or not urgent
|
|
||||||
Urgent(String, bool),
|
|
||||||
/// Declares the item with `app_id` has been hovered over
|
/// Declares the item with `app_id` has been hovered over
|
||||||
Hover(String),
|
Hover(String),
|
||||||
}
|
}
|
||||||
|
@ -61,7 +57,7 @@ pub enum LauncherUpdate {
|
||||||
#[derive(Debug)]
|
#[derive(Debug)]
|
||||||
pub enum ItemEvent {
|
pub enum ItemEvent {
|
||||||
FocusItem(String),
|
FocusItem(String),
|
||||||
FocusWindow(i64),
|
FocusWindow(usize),
|
||||||
OpenItem(String),
|
OpenItem(String),
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -100,24 +96,26 @@ impl Module<gtk::Box> for LauncherModule {
|
||||||
|
|
||||||
let items = Arc::new(Mutex::new(items));
|
let items = Arc::new(Mutex::new(items));
|
||||||
|
|
||||||
let open_windows = await_sync(async {
|
|
||||||
let sway = get_client().await;
|
|
||||||
let mut sway = sway.lock().await;
|
|
||||||
get_open_windows(&mut sway).await
|
|
||||||
})?;
|
|
||||||
|
|
||||||
{
|
{
|
||||||
let mut items = items.lock().expect("Failed to get lock on items");
|
let items = Arc::clone(&items);
|
||||||
for window in open_windows {
|
let tx = tx.clone();
|
||||||
let id = get_node_id(&window).to_string();
|
spawn(async move {
|
||||||
|
let wl = wayland::get_client().await;
|
||||||
|
let open_windows = wl
|
||||||
|
.toplevels
|
||||||
|
.read()
|
||||||
|
.expect("Failed to get read lock on toplevels");
|
||||||
|
|
||||||
let item = items.get_mut(&id);
|
let mut items = items.lock().expect("Failed to get lock on items");
|
||||||
|
|
||||||
|
for (window, _) in open_windows.clone().into_iter() {
|
||||||
|
let item = items.get_mut(&window.app_id);
|
||||||
match item {
|
match item {
|
||||||
Some(item) => {
|
Some(item) => {
|
||||||
item.merge_node(window);
|
item.merge_toplevel(window);
|
||||||
}
|
}
|
||||||
None => {
|
None => {
|
||||||
items.insert(id, window.into());
|
items.insert(window.app_id.clone(), window.into());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -128,41 +126,43 @@ impl Module<gtk::Box> for LauncherModule {
|
||||||
item.clone(),
|
item.clone(),
|
||||||
)))?;
|
)))?;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Ok::<(), Report>(())
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
let items2 = Arc::clone(&items);
|
let items2 = Arc::clone(&items);
|
||||||
spawn(async move {
|
spawn(async move {
|
||||||
let items = items2;
|
let items = items2;
|
||||||
|
|
||||||
let mut srx = {
|
let mut wlrx = {
|
||||||
let sway = get_sub_client();
|
let wl = wayland::get_client().await;
|
||||||
sway.subscribe_window()
|
wl.subscribe_toplevels()
|
||||||
};
|
};
|
||||||
|
|
||||||
while let Ok(event) = srx.recv().await {
|
let send_update = |update: LauncherUpdate| tx.send(ModuleUpdateEvent::Update(update));
|
||||||
|
|
||||||
|
while let Ok(event) = wlrx.recv().await {
|
||||||
trace!("event: {:?}", event);
|
trace!("event: {:?}", event);
|
||||||
|
|
||||||
let window = event.container;
|
let window = event.toplevel;
|
||||||
let id = get_node_id(&window).to_string();
|
let app_id = window.app_id.clone();
|
||||||
|
|
||||||
let send_update =
|
|
||||||
|update: LauncherUpdate| tx.send(ModuleUpdateEvent::Update(update));
|
|
||||||
|
|
||||||
let items = || items.lock().expect("Failed to get lock on items");
|
let items = || items.lock().expect("Failed to get lock on items");
|
||||||
|
|
||||||
match event.change {
|
match event.change {
|
||||||
WindowChange::New => {
|
ToplevelChange::New => {
|
||||||
let new_item = {
|
let new_item = {
|
||||||
let mut items = items();
|
let mut items = items();
|
||||||
match items.get_mut(&id) {
|
match items.get_mut(&app_id) {
|
||||||
None => {
|
None => {
|
||||||
let item: Item = window.into();
|
let item: Item = window.into();
|
||||||
items.insert(id.clone(), item.clone());
|
items.insert(app_id.clone(), item.clone());
|
||||||
|
|
||||||
ItemOrWindow::Item(item)
|
ItemOrWindow::Item(item)
|
||||||
}
|
}
|
||||||
Some(item) => {
|
Some(item) => {
|
||||||
let window = item.merge_node(window);
|
let window = item.merge_toplevel(window);
|
||||||
ItemOrWindow::Window(window)
|
ItemOrWindow::Window(window)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -173,19 +173,19 @@ impl Module<gtk::Box> for LauncherModule {
|
||||||
send_update(LauncherUpdate::AddItem(item)).await
|
send_update(LauncherUpdate::AddItem(item)).await
|
||||||
}
|
}
|
||||||
ItemOrWindow::Window(window) => {
|
ItemOrWindow::Window(window) => {
|
||||||
send_update(LauncherUpdate::AddWindow(id, window)).await
|
send_update(LauncherUpdate::AddWindow(app_id, window)).await
|
||||||
}
|
}
|
||||||
}?;
|
}?;
|
||||||
}
|
}
|
||||||
WindowChange::Close => {
|
ToplevelChange::Close => {
|
||||||
let remove_item = {
|
let remove_item = {
|
||||||
let mut items = items();
|
let mut items = items();
|
||||||
match items.get_mut(&id) {
|
match items.get_mut(&app_id) {
|
||||||
Some(item) => {
|
Some(item) => {
|
||||||
item.unmerge_node(&window);
|
item.unmerge_toplevel(&window);
|
||||||
|
|
||||||
if item.windows.is_empty() {
|
if item.windows.is_empty() {
|
||||||
items.remove(&id);
|
items.remove(&app_id);
|
||||||
Some(ItemOrWindowId::Item)
|
Some(ItemOrWindowId::Item)
|
||||||
} else {
|
} else {
|
||||||
Some(ItemOrWindowId::Window)
|
Some(ItemOrWindowId::Window)
|
||||||
|
@ -197,58 +197,47 @@ impl Module<gtk::Box> for LauncherModule {
|
||||||
|
|
||||||
match remove_item {
|
match remove_item {
|
||||||
Some(ItemOrWindowId::Item) => {
|
Some(ItemOrWindowId::Item) => {
|
||||||
send_update(LauncherUpdate::RemoveItem(id)).await?;
|
send_update(LauncherUpdate::RemoveItem(app_id)).await?;
|
||||||
}
|
}
|
||||||
Some(ItemOrWindowId::Window) => {
|
Some(ItemOrWindowId::Window) => {
|
||||||
send_update(LauncherUpdate::RemoveWindow(id, window.id)).await?;
|
send_update(LauncherUpdate::RemoveWindow(app_id, window.id))
|
||||||
|
.await?;
|
||||||
}
|
}
|
||||||
None => {}
|
None => {}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
WindowChange::Focus => {
|
ToplevelChange::Focus(focused) => {
|
||||||
let prev_id = {
|
let update_title = if focused {
|
||||||
let mut items = items();
|
if let Some(item) = items().get_mut(&app_id) {
|
||||||
|
|
||||||
let prev_focused =
|
|
||||||
items.iter_mut().find(|item| item.open_state.is_focused());
|
|
||||||
if let Some(prev_focused) = prev_focused {
|
|
||||||
prev_focused.set_unfocused();
|
|
||||||
Some(prev_focused.app_id.to_string())
|
|
||||||
} else {
|
|
||||||
None
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let mut update_title = false;
|
|
||||||
if let Some(item) = items().get_mut(&id) {
|
|
||||||
item.set_window_focused(window.id, true);
|
item.set_window_focused(window.id, true);
|
||||||
|
|
||||||
// might be switching focus between windows of same app
|
// might be switching focus between windows of same app
|
||||||
if item.windows.len() > 1 {
|
if item.windows.len() > 1 {
|
||||||
item.set_window_name(window.id, window.name.clone());
|
item.set_window_name(window.id, window.title.clone());
|
||||||
update_title = true;
|
true
|
||||||
|
} else {
|
||||||
|
false
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
false
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
};
|
||||||
|
|
||||||
send_update(LauncherUpdate::Focus(id.clone(), prev_id)).await?;
|
send_update(LauncherUpdate::Focus(app_id.clone(), focused)).await?;
|
||||||
|
|
||||||
if update_title {
|
if update_title {
|
||||||
send_update(LauncherUpdate::Title(id, window.id, window.name)).await?;
|
send_update(LauncherUpdate::Title(app_id, window.id, window.title))
|
||||||
|
.await?;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
WindowChange::Title => {
|
ToplevelChange::Title(title) => {
|
||||||
if let Some(item) = items().get_mut(&id) {
|
if let Some(item) = items().get_mut(&app_id) {
|
||||||
item.set_window_name(window.id, window.name.clone());
|
item.set_window_name(window.id, title.clone());
|
||||||
}
|
}
|
||||||
|
|
||||||
send_update(LauncherUpdate::Title(id, window.id, window.name)).await?;
|
send_update(LauncherUpdate::Title(app_id, window.id, title)).await?;
|
||||||
}
|
|
||||||
WindowChange::Urgent => {
|
|
||||||
if let Some(item) = items().get_mut(&id) {
|
|
||||||
item.set_window_urgent(window.id, window.urgent);
|
|
||||||
}
|
|
||||||
|
|
||||||
send_update(LauncherUpdate::Urgent(id, window.urgent)).await?;
|
|
||||||
}
|
}
|
||||||
_ => {}
|
_ => {}
|
||||||
}
|
}
|
||||||
|
@ -259,8 +248,6 @@ impl Module<gtk::Box> for LauncherModule {
|
||||||
|
|
||||||
// listen to ui events
|
// listen to ui events
|
||||||
spawn(async move {
|
spawn(async move {
|
||||||
let sway = get_client().await;
|
|
||||||
|
|
||||||
while let Some(event) = rx.recv().await {
|
while let Some(event) = rx.recv().await {
|
||||||
trace!("{:?}", event);
|
trace!("{:?}", event);
|
||||||
|
|
||||||
|
@ -287,25 +274,26 @@ impl Module<gtk::Box> for LauncherModule {
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
let selector = {
|
let wl = wayland::get_client().await;
|
||||||
let items = items.lock().expect("Failed to get lock on items");
|
let items = items.lock().expect("Failed to get lock on items");
|
||||||
|
|
||||||
match event {
|
let id = match event {
|
||||||
ItemEvent::FocusItem(app_id) => items.get(&app_id).map(|item| {
|
ItemEvent::FocusItem(app_id) => items
|
||||||
if item.is_xwayland {
|
.get(&app_id)
|
||||||
format!("[class={}]", app_id)
|
.and_then(|item| item.windows.first().map(|win| win.id)),
|
||||||
} else {
|
ItemEvent::FocusWindow(id) => Some(id),
|
||||||
format!("[app_id={}]", app_id)
|
|
||||||
}
|
|
||||||
}),
|
|
||||||
ItemEvent::FocusWindow(con_id) => Some(format!("[con_id={}]", con_id)),
|
|
||||||
ItemEvent::OpenItem(_) => unreachable!(),
|
ItemEvent::OpenItem(_) => unreachable!(),
|
||||||
}
|
|
||||||
};
|
};
|
||||||
|
|
||||||
if let Some(selector) = selector {
|
if let Some(id) = id {
|
||||||
let mut sway = sway.lock().await;
|
let toplevels = wl
|
||||||
sway.run_command(format!("{} focus", selector)).await?;
|
.toplevels
|
||||||
|
.read()
|
||||||
|
.expect("Failed to get read lock on toplevels");
|
||||||
|
let seat = wl.seats.first().unwrap();
|
||||||
|
if let Some((_top, handle)) = toplevels.get(&id) {
|
||||||
|
handle.activate(seat);
|
||||||
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -391,20 +379,11 @@ impl Module<gtk::Box> for LauncherModule {
|
||||||
menu_state.num_windows -= 1;
|
menu_state.num_windows -= 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
LauncherUpdate::Focus(new, prev) => {
|
LauncherUpdate::Focus(app_id, focus) => {
|
||||||
debug!(
|
debug!("Changing focus to {} on item with id {}", focus, app_id);
|
||||||
"Changing focus to item with id {} (removing from {:?})",
|
|
||||||
new, prev
|
|
||||||
);
|
|
||||||
|
|
||||||
if let Some(prev) = prev {
|
if let Some(button) = buttons.get(&app_id) {
|
||||||
if let Some(button) = buttons.get(&prev) {
|
button.set_focused(focus);
|
||||||
button.set_focused(false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if let Some(button) = buttons.get(&new) {
|
|
||||||
button.set_focused(true);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
LauncherUpdate::Title(app_id, _, name) => {
|
LauncherUpdate::Title(app_id, _, name) => {
|
||||||
|
@ -412,17 +391,10 @@ impl Module<gtk::Box> for LauncherModule {
|
||||||
|
|
||||||
if show_names {
|
if show_names {
|
||||||
if let Some(button) = buttons.get(&app_id) {
|
if let Some(button) = buttons.get(&app_id) {
|
||||||
button.button.set_label(&name.unwrap_or_default());
|
button.button.set_label(&name);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
LauncherUpdate::Urgent(app_id, urgent) => {
|
|
||||||
debug!("Updating urgency for item with id {}: {}", app_id, urgent);
|
|
||||||
|
|
||||||
if let Some(button) = buttons.get(&app_id) {
|
|
||||||
button.set_urgent(urgent);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
LauncherUpdate::Hover(_) => {}
|
LauncherUpdate::Hover(_) => {}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -444,7 +416,7 @@ impl Module<gtk::Box> for LauncherModule {
|
||||||
) -> Option<gtk::Box> {
|
) -> Option<gtk::Box> {
|
||||||
let container = gtk::Box::new(Orientation::Vertical, 0);
|
let container = gtk::Box::new(Orientation::Vertical, 0);
|
||||||
|
|
||||||
let mut buttons = Collection::<String, Collection<i64, Button>>::new();
|
let mut buttons = Collection::<String, Collection<usize, Button>>::new();
|
||||||
|
|
||||||
{
|
{
|
||||||
let container = container.clone();
|
let container = container.clone();
|
||||||
|
@ -458,7 +430,7 @@ impl Module<gtk::Box> for LauncherModule {
|
||||||
.into_iter()
|
.into_iter()
|
||||||
.map(|win| {
|
.map(|win| {
|
||||||
let button = Button::builder()
|
let button = Button::builder()
|
||||||
.label(win.name.as_ref().unwrap_or(&String::new()))
|
.label(&win.name)
|
||||||
.height_request(40)
|
.height_request(40)
|
||||||
.width_request(100)
|
.width_request(100)
|
||||||
.build();
|
.build();
|
||||||
|
@ -484,7 +456,7 @@ impl Module<gtk::Box> for LauncherModule {
|
||||||
LauncherUpdate::AddWindow(app_id, win) => {
|
LauncherUpdate::AddWindow(app_id, win) => {
|
||||||
if let Some(buttons) = buttons.get_mut(&app_id) {
|
if let Some(buttons) = buttons.get_mut(&app_id) {
|
||||||
let button = Button::builder()
|
let button = Button::builder()
|
||||||
.label(win.name.as_ref().unwrap_or(&String::new()))
|
.label(&win.name)
|
||||||
.height_request(40)
|
.height_request(40)
|
||||||
.width_request(100)
|
.width_request(100)
|
||||||
.build();
|
.build();
|
||||||
|
@ -512,12 +484,10 @@ impl Module<gtk::Box> for LauncherModule {
|
||||||
LauncherUpdate::Title(app_id, win_id, title) => {
|
LauncherUpdate::Title(app_id, win_id, title) => {
|
||||||
if let Some(buttons) = buttons.get_mut(&app_id) {
|
if let Some(buttons) = buttons.get_mut(&app_id) {
|
||||||
if let Some(button) = buttons.get(&win_id) {
|
if let Some(button) = buttons.get(&win_id) {
|
||||||
if let Some(title) = title {
|
|
||||||
button.set_label(&title);
|
button.set_label(&title);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
LauncherUpdate::Hover(app_id) => {
|
LauncherUpdate::Hover(app_id) => {
|
||||||
// empty current buttons
|
// empty current buttons
|
||||||
for child in container.children() {
|
for child in container.children() {
|
||||||
|
|
|
@ -1,35 +1,23 @@
|
||||||
use swayipc_async::Node;
|
use crate::wayland::ToplevelInfo;
|
||||||
|
|
||||||
/// Open state for a launcher item, or item window.
|
/// Open state for a launcher item, or item window.
|
||||||
#[derive(Debug, Clone, Eq, PartialEq, Copy)]
|
#[derive(Debug, Clone, Eq, PartialEq, Copy)]
|
||||||
pub enum OpenState {
|
pub enum OpenState {
|
||||||
Closed,
|
Closed,
|
||||||
Open { focused: bool, urgent: bool },
|
Open { focused: bool },
|
||||||
}
|
}
|
||||||
|
|
||||||
impl OpenState {
|
impl OpenState {
|
||||||
/// Creates from `SwayNode`
|
/// Creates from `SwayNode`
|
||||||
pub const fn from_node(node: &Node) -> Self {
|
pub const fn from_toplevel(toplevel: &ToplevelInfo) -> Self {
|
||||||
Self::Open {
|
Self::Open {
|
||||||
focused: node.focused,
|
focused: toplevel.active,
|
||||||
urgent: node.urgent,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Creates open with focused
|
/// Creates open with focused
|
||||||
pub const fn focused(focused: bool) -> Self {
|
pub const fn focused(focused: bool) -> Self {
|
||||||
Self::Open {
|
Self::Open { focused }
|
||||||
focused,
|
|
||||||
urgent: false,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/// Creates open with urgent
|
|
||||||
pub const fn urgent(urgent: bool) -> Self {
|
|
||||||
Self::Open {
|
|
||||||
focused: false,
|
|
||||||
urgent,
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Checks if open
|
/// Checks if open
|
||||||
|
@ -39,12 +27,7 @@ impl OpenState {
|
||||||
|
|
||||||
/// Checks if open with focus
|
/// Checks if open with focus
|
||||||
pub const fn is_focused(self) -> bool {
|
pub const fn is_focused(self) -> bool {
|
||||||
matches!(self, Self::Open { focused: true, .. })
|
matches!(self, Self::Open { focused: true })
|
||||||
}
|
|
||||||
|
|
||||||
/// check if open with urgent
|
|
||||||
pub const fn is_urgent(self) -> bool {
|
|
||||||
matches!(self, Self::Open { urgent: true, .. })
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Merges states together to produce a single state.
|
/// Merges states together to produce a single state.
|
||||||
|
@ -56,7 +39,6 @@ impl OpenState {
|
||||||
if merged.is_open() || current.is_open() {
|
if merged.is_open() || current.is_open() {
|
||||||
Self::Open {
|
Self::Open {
|
||||||
focused: merged.is_focused() || current.is_focused(),
|
focused: merged.is_focused() || current.is_focused(),
|
||||||
urgent: merged.is_urgent() || current.is_urgent(),
|
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Self::Closed
|
Self::Closed
|
||||||
|
|
|
@ -1,21 +1,27 @@
|
||||||
use std::sync::{Arc, RwLock};
|
|
||||||
use super::{Env, ToplevelHandler};
|
use super::{Env, ToplevelHandler};
|
||||||
|
use crate::collection::Collection;
|
||||||
|
use crate::wayland::toplevel::{ToplevelEvent, ToplevelInfo};
|
||||||
use crate::wayland::toplevel_manager::listen_for_toplevels;
|
use crate::wayland::toplevel_manager::listen_for_toplevels;
|
||||||
|
use crate::wayland::ToplevelChange;
|
||||||
use smithay_client_toolkit::environment::Environment;
|
use smithay_client_toolkit::environment::Environment;
|
||||||
use smithay_client_toolkit::output::{with_output_info, OutputInfo};
|
use smithay_client_toolkit::output::{with_output_info, OutputInfo};
|
||||||
use smithay_client_toolkit::reexports::calloop;
|
use smithay_client_toolkit::reexports::calloop;
|
||||||
use smithay_client_toolkit::{new_default_environment, WaylandSource};
|
use smithay_client_toolkit::{new_default_environment, WaylandSource};
|
||||||
|
use std::sync::{Arc, RwLock};
|
||||||
|
use std::time::Duration;
|
||||||
use tokio::sync::{broadcast, oneshot};
|
use tokio::sync::{broadcast, oneshot};
|
||||||
use tokio::task::spawn_blocking;
|
use tokio::task::spawn_blocking;
|
||||||
use tracing::{trace};
|
use tracing::trace;
|
||||||
use wayland_protocols::wlr::unstable::foreign_toplevel::v1::client::zwlr_foreign_toplevel_manager_v1::ZwlrForeignToplevelManagerV1;
|
use wayland_protocols::wlr::unstable::foreign_toplevel::v1::client::{
|
||||||
use crate::collection::Collection;
|
zwlr_foreign_toplevel_handle_v1::ZwlrForeignToplevelHandleV1,
|
||||||
use crate::wayland::toplevel::{ToplevelEvent, ToplevelInfo};
|
zwlr_foreign_toplevel_manager_v1::ZwlrForeignToplevelManagerV1,
|
||||||
use crate::wayland::ToplevelChange;
|
};
|
||||||
|
use wayland_client::protocol::wl_seat::WlSeat;
|
||||||
|
|
||||||
pub struct WaylandClient {
|
pub struct WaylandClient {
|
||||||
pub outputs: Vec<OutputInfo>,
|
pub outputs: Vec<OutputInfo>,
|
||||||
pub toplevels: Arc<RwLock<Collection<String, ToplevelInfo>>>,
|
pub seats: Vec<WlSeat>,
|
||||||
|
pub toplevels: Arc<RwLock<Collection<usize, (ToplevelInfo, ZwlrForeignToplevelHandleV1)>>>,
|
||||||
toplevel_tx: broadcast::Sender<ToplevelEvent>,
|
toplevel_tx: broadcast::Sender<ToplevelEvent>,
|
||||||
_toplevel_rx: broadcast::Receiver<ToplevelEvent>,
|
_toplevel_rx: broadcast::Receiver<ToplevelEvent>,
|
||||||
}
|
}
|
||||||
|
@ -23,6 +29,7 @@ pub struct WaylandClient {
|
||||||
impl WaylandClient {
|
impl WaylandClient {
|
||||||
pub(super) async fn new() -> Self {
|
pub(super) async fn new() -> Self {
|
||||||
let (output_tx, output_rx) = oneshot::channel();
|
let (output_tx, output_rx) = oneshot::channel();
|
||||||
|
let (seat_tx, seat_rx) = oneshot::channel();
|
||||||
let (toplevel_tx, toplevel_rx) = broadcast::channel(32);
|
let (toplevel_tx, toplevel_rx) = broadcast::channel(32);
|
||||||
|
|
||||||
let toplevel_tx2 = toplevel_tx.clone();
|
let toplevel_tx2 = toplevel_tx.clone();
|
||||||
|
@ -41,21 +48,24 @@ impl WaylandClient {
|
||||||
.send(outputs)
|
.send(outputs)
|
||||||
.expect("Failed to send outputs out of task");
|
.expect("Failed to send outputs out of task");
|
||||||
|
|
||||||
|
let seats = env.get_all_seats();
|
||||||
|
seat_tx.send(seats.into_iter().map(|seat| seat.detach()).collect::<Vec<WlSeat>>()).expect("Failed to send seats out of task");
|
||||||
|
|
||||||
let _toplevel_manager = env.require_global::<ZwlrForeignToplevelManagerV1>();
|
let _toplevel_manager = env.require_global::<ZwlrForeignToplevelManagerV1>();
|
||||||
|
|
||||||
let _listener = listen_for_toplevels(env, move |_handle, event, _ddata| {
|
let _listener = listen_for_toplevels(env, move |handle, event, _ddata| {
|
||||||
trace!("Received toplevel event: {:?}", event);
|
trace!("Received toplevel event: {:?}", event);
|
||||||
|
|
||||||
if event.change != ToplevelChange::Close {
|
if event.change != ToplevelChange::Close {
|
||||||
toplevels2
|
toplevels2
|
||||||
.write()
|
.write()
|
||||||
.expect("Failed to get write lock on toplevels")
|
.expect("Failed to get write lock on toplevels")
|
||||||
.insert(event.toplevel.app_id.clone(), event.toplevel.clone());
|
.insert(event.toplevel.id, (event.toplevel.clone(), handle));
|
||||||
} else {
|
} else {
|
||||||
toplevels2
|
toplevels2
|
||||||
.write()
|
.write()
|
||||||
.expect("Failed to get write lock on toplevels")
|
.expect("Failed to get write lock on toplevels")
|
||||||
.remove(&event.toplevel.app_id);
|
.remove(&event.toplevel.id);
|
||||||
}
|
}
|
||||||
|
|
||||||
toplevel_tx2
|
toplevel_tx2
|
||||||
|
@ -69,7 +79,9 @@ impl WaylandClient {
|
||||||
.unwrap();
|
.unwrap();
|
||||||
|
|
||||||
loop {
|
loop {
|
||||||
event_loop.dispatch(None, &mut ()).unwrap();
|
// TODO: Avoid need for duration here - can we force some event when sending requests?
|
||||||
|
event_loop.dispatch(Duration::from_millis(50), &mut ()).unwrap();
|
||||||
|
event_loop.
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -77,16 +89,11 @@ impl WaylandClient {
|
||||||
.await
|
.await
|
||||||
.expect("Failed to receive outputs from task");
|
.expect("Failed to receive outputs from task");
|
||||||
|
|
||||||
// spawn(async move {
|
let seats = seat_rx.await.expect("Failed to receive seats from task");
|
||||||
// println!("start");
|
|
||||||
// while let Ok(ev) = toplevel_rx.recv().await {
|
|
||||||
// println!("recv {:?}", ev)
|
|
||||||
// }
|
|
||||||
// println!("stop");
|
|
||||||
// });
|
|
||||||
|
|
||||||
Self {
|
Self {
|
||||||
outputs,
|
outputs,
|
||||||
|
seats,
|
||||||
toplevels,
|
toplevels,
|
||||||
toplevel_tx,
|
toplevel_tx,
|
||||||
_toplevel_rx: toplevel_rx,
|
_toplevel_rx: toplevel_rx,
|
||||||
|
|
|
@ -1,13 +1,18 @@
|
||||||
use std::collections::HashSet;
|
use std::collections::HashSet;
|
||||||
use std::sync::{Arc, RwLock};
|
use std::sync::{Arc, RwLock};
|
||||||
|
use std::sync::atomic::{AtomicUsize, Ordering};
|
||||||
use wayland_client::{DispatchData, Main};
|
use wayland_client::{DispatchData, Main};
|
||||||
use wayland_protocols::wlr::unstable::foreign_toplevel::v1::client::zwlr_foreign_toplevel_handle_v1::{Event, ZwlrForeignToplevelHandleV1};
|
use wayland_protocols::wlr::unstable::foreign_toplevel::v1::client::zwlr_foreign_toplevel_handle_v1::{Event, ZwlrForeignToplevelHandleV1};
|
||||||
|
|
||||||
const STATE_ACTIVE: u32 = 2;
|
const STATE_ACTIVE: u32 = 2;
|
||||||
const STATE_FULLSCREEN: u32 = 3;
|
const STATE_FULLSCREEN: u32 = 3;
|
||||||
|
|
||||||
|
static COUNTER: AtomicUsize = AtomicUsize::new(1);
|
||||||
|
fn get_id() -> usize { COUNTER.fetch_add(1, Ordering::Relaxed) }
|
||||||
|
|
||||||
#[derive(Debug, Clone, Default)]
|
#[derive(Debug, Clone, Default)]
|
||||||
pub struct ToplevelInfo {
|
pub struct ToplevelInfo {
|
||||||
|
pub id: usize,
|
||||||
pub app_id: String,
|
pub app_id: String,
|
||||||
pub title: String,
|
pub title: String,
|
||||||
pub active: bool,
|
pub active: bool,
|
||||||
|
@ -16,6 +21,13 @@ pub struct ToplevelInfo {
|
||||||
ready: bool,
|
ready: bool,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl ToplevelInfo {
|
||||||
|
fn new() -> Self {
|
||||||
|
let id = get_id();
|
||||||
|
Self { id, ..Default::default() }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub struct Toplevel;
|
pub struct Toplevel;
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
|
@ -112,7 +124,7 @@ impl Toplevel {
|
||||||
where
|
where
|
||||||
F: FnMut(ToplevelEvent, DispatchData) + 'static,
|
F: FnMut(ToplevelEvent, DispatchData) + 'static,
|
||||||
{
|
{
|
||||||
let inner = Arc::new(RwLock::new(ToplevelInfo::default()));
|
let inner = Arc::new(RwLock::new(ToplevelInfo::new()));
|
||||||
|
|
||||||
handle.quick_assign(move |_handle, event, ddata| {
|
handle.quick_assign(move |_handle, event, ddata| {
|
||||||
let mut inner = inner
|
let mut inner = inner
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue