// Copyright (C) 2021-2022 Tassilo Horn // // This program is free software: you can redistribute it and/or modify it // under the terms of the GNU General Public License as published by the Free // Software Foundation, either version 3 of the License, or (at your option) // any later version. // // This program is distributed in the hope that it will be useful, but WITHOUT // ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or // FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for // more details. // // You should have received a copy of the GNU General Public License along with // this program. If not, see . //! Convenience data structures built from the IPC structs. use crate::config; use crate::util; use crate::util::DisplayFormat; use lazy_static::lazy_static; use serde::{Deserialize, Serialize}; use std::cell::RefCell; use std::cmp; use std::collections::HashMap; use std::rc::Rc; use swayipc as s; pub fn get_root_node(include_scratchpad: bool) -> s::Node { match s::Connection::new() { Ok(mut con) => { let mut root = con.get_tree().expect("Got no root node"); if !include_scratchpad { root.nodes.retain(|o| !o.is_scratchpad()); } root } Err(err) => panic!("{}", err), } } /// Immutable Node Iterator /// /// Iterates nodes in depth-first order, tiled nodes before floating nodes. pub struct NodeIter<'a> { stack: Vec<&'a s::Node>, } impl<'a> NodeIter<'a> { pub fn new(node: &'a s::Node) -> NodeIter { NodeIter { stack: vec![node] } } } impl<'a> Iterator for NodeIter<'a> { type Item = &'a s::Node; fn next(&mut self) -> Option { if let Some(node) = self.stack.pop() { for n in &node.floating_nodes { self.stack.push(n); } for n in &node.nodes { self.stack.push(n); } Some(node) } else { None } } } #[derive(Debug, PartialEq, Eq, Clone)] pub enum Type { Root, Output, Workspace, Container, Window, } /// Extension methods for [`swayipc::Node`]. pub trait NodeMethods { fn iter(&self) -> NodeIter; fn get_type(&self) -> Type; fn get_app_name(&self) -> &str; fn nodes_of_type(&self, t: Type) -> Vec<&s::Node>; fn get_name(&self) -> &str; fn is_scratchpad(&self) -> bool; fn is_floating(&self) -> bool; fn is_current(&self) -> bool; } impl NodeMethods for s::Node { fn iter(&self) -> NodeIter { NodeIter::new(self) } fn get_type(&self) -> Type { match self.node_type { s::NodeType::Root => Type::Root, s::NodeType::Output => Type::Output, s::NodeType::Workspace => Type::Workspace, s::NodeType::FloatingCon => Type::Window, _ => { if self.node_type == s::NodeType::Con && self.name.is_none() && self.app_id.is_none() && self.pid.is_none() && self.shell.is_none() && self.window_properties.is_none() && self.layout != s::NodeLayout::None { Type::Container } else if (self.node_type == s::NodeType::Con || self.node_type == s::NodeType::FloatingCon) // Apparently there can be windows without app_id, name, // and window_properties.class, e.g., dolphin-emu-nogui. && self.pid.is_some() // FIXME: While technically correct, old sway versions (up to // at least sway-1.4) don't expose shell in IPC. So comment in // again when all major distros have a recent enough sway // package. //&& self.shell.is_some() { Type::Window } else { panic!( "Don't know type of node with id {} and node_type {:?}\n{:?}", self.id, self.node_type, self ) } } } } fn get_name(&self) -> &str { if let Some(name) = &self.name { name.as_ref() } else { "" } } fn get_app_name(&self) -> &str { if let Some(app_id) = &self.app_id { app_id } else if let Some(wp_class) = self .window_properties .as_ref() .and_then(|wp| wp.class.as_ref()) { wp_class } else { "" } } fn is_scratchpad(&self) -> bool { let name = self.get_name(); name.eq("__i3") || name.eq("__i3_scratch") } fn nodes_of_type(&self, t: Type) -> Vec<&s::Node> { self.iter().filter(|n| n.get_type() == t).collect() } fn is_floating(&self) -> bool { self.node_type == s::NodeType::FloatingCon } fn is_current(&self) -> bool { self.iter().any(|n| n.focused) } } /// Extra properties gathered by swayrd for windows and workspaces. #[derive(Copy, Clone, Debug, Deserialize, Serialize)] pub struct ExtraProps { pub last_focus_tick: u64, pub last_focus_tick_for_next_prev_seq: u64, } pub struct Tree<'a> { root: &'a s::Node, id_node: HashMap, id_parent: HashMap, extra_props: &'a HashMap, } #[derive(Copy, Clone, PartialEq, Eq)] enum IndentLevel { Fixed(usize), WorkspacesZeroWindowsOne, TreeDepth(usize), } pub struct DisplayNode<'a> { pub node: &'a s::Node, pub tree: &'a Tree<'a>, indent_level: IndentLevel, } impl<'a> Tree<'a> { fn get_node_by_id(&self, id: i64) -> &&s::Node { self.id_node .get(&id) .unwrap_or_else(|| panic!("No node with id {}", id)) } fn get_parent_node(&self, id: i64) -> Option<&&s::Node> { self.id_parent.get(&id).map(|pid| self.get_node_by_id(*pid)) } pub fn get_parent_node_of_type( &self, id: i64, t: Type, ) -> Option<&&s::Node> { let n = self.get_node_by_id(id); if n.get_type() == t { Some(n) } else if let Some(pid) = self.id_parent.get(&id) { self.get_parent_node_of_type(*pid, t) } else { None } } pub fn last_focus_tick(&self, id: i64) -> u64 { self.extra_props.get(&id).map_or(0, |wp| wp.last_focus_tick) } pub fn last_focus_tick_for_next_prev_seq(&self, id: i64) -> u64 { self.extra_props .get(&id) .map_or(0, |wp| wp.last_focus_tick_for_next_prev_seq) } fn sorted_nodes_of_type_1( &self, node: &'a s::Node, t: Type, ) -> Vec<&s::Node> { let mut v: Vec<&s::Node> = node.nodes_of_type(t); self.sort_by_urgency_and_lru_time_1(&mut v); v } fn sorted_nodes_of_type(&self, t: Type) -> Vec<&s::Node> { self.sorted_nodes_of_type_1(self.root, t) } fn as_display_nodes( &self, v: &[&'a s::Node], indent_level: IndentLevel, ) -> Vec { v.iter() .map(|node| DisplayNode { node, tree: self, indent_level, }) .collect() } pub fn get_current_workspace(&self) -> &s::Node { self.root .iter() .find(|n| n.get_type() == Type::Workspace && n.is_current()) .expect("No current Workspace") } pub fn get_outputs(&self) -> Vec { let outputs: Vec<&s::Node> = self .root .iter() .filter(|n| n.get_type() == Type::Output && !n.is_scratchpad()) .collect(); self.as_display_nodes(&outputs, IndentLevel::Fixed(0)) } pub fn get_workspaces(&self) -> Vec { let mut v = self.sorted_nodes_of_type(Type::Workspace); if !v.is_empty() { v.rotate_left(1); } self.as_display_nodes(&v, IndentLevel::Fixed(0)) } pub fn get_windows(&self) -> Vec { let mut v = self.sorted_nodes_of_type(Type::Window); // Rotate, but only non-urgent windows. Those should stay at the front // as they are the most likely switch candidates. let mut x; if !v.is_empty() { x = vec![]; loop { if !v.is_empty() && v[0].urgent { x.push(v.remove(0)); } else { break; } } if !v.is_empty() { v.rotate_left(1); x.append(&mut v); } } else { x = v; } self.as_display_nodes(&x, IndentLevel::Fixed(0)) } pub fn get_workspaces_and_windows(&self) -> Vec { let workspaces = self.sorted_nodes_of_type(Type::Workspace); let mut first = true; let mut v = vec![]; for ws in workspaces { v.push(ws); let mut wins = self.sorted_nodes_of_type_1(ws, Type::Window); if first && !wins.is_empty() { wins.rotate_left(1); first = false; } v.append(&mut wins); } self.as_display_nodes(&v, IndentLevel::WorkspacesZeroWindowsOne) } fn sort_by_urgency_and_lru_time_1(&self, v: &mut Vec<&s::Node>) { v.sort_by(|a, b| { if a.urgent && !b.urgent { cmp::Ordering::Less } else if !a.urgent && b.urgent { cmp::Ordering::Greater } else { let lru_a = self.last_focus_tick(a.id); let lru_b = self.last_focus_tick(b.id); lru_a.cmp(&lru_b).reverse() } }); } fn push_subtree_sorted( &self, n: &'a s::Node, v: Rc>>, ) { v.borrow_mut().push(n); let mut children: Vec<&s::Node> = n.nodes.iter().collect(); children.append(&mut n.floating_nodes.iter().collect()); self.sort_by_urgency_and_lru_time_1(&mut children); for c in children { self.push_subtree_sorted(c, Rc::clone(&v)); } } pub fn get_outputs_workspaces_containers_and_windows( &self, ) -> Vec { let outputs = self.sorted_nodes_of_type(Type::Output); let v: Rc>> = Rc::new(RefCell::new(vec![])); for o in outputs { self.push_subtree_sorted(o, Rc::clone(&v)); } let x = self.as_display_nodes(&*v.borrow(), IndentLevel::TreeDepth(1)); x } pub fn get_workspaces_containers_and_windows(&self) -> Vec { let workspaces = self.sorted_nodes_of_type(Type::Workspace); let v: Rc>> = Rc::new(RefCell::new(vec![])); for ws in workspaces { self.push_subtree_sorted(ws, Rc::clone(&v)); } let x = self.as_display_nodes(&*v.borrow(), IndentLevel::TreeDepth(2)); x } pub fn is_child_of_tiled_container(&self, id: i64) -> bool { match self.get_parent_node(id) { Some(n) => { n.layout == s::NodeLayout::SplitH || n.layout == s::NodeLayout::SplitV } None => false, } } pub fn is_child_of_tabbed_or_stacked_container(&self, id: i64) -> bool { match self.get_parent_node(id) { Some(n) => { n.layout == s::NodeLayout::Tabbed || n.layout == s::NodeLayout::Stacked } None => false, } } } fn init_id_parent<'a>( n: &'a s::Node, parent: Option<&'a s::Node>, id_node: &mut HashMap, id_parent: &mut HashMap, ) { id_node.insert(n.id, n); if let Some(p) = parent { id_parent.insert(n.id, p.id); } for c in &n.nodes { init_id_parent(c, Some(n), id_node, id_parent); } for c in &n.floating_nodes { init_id_parent(c, Some(n), id_node, id_parent); } } pub fn get_tree<'a>( root: &'a s::Node, extra_props: &'a HashMap, ) -> Tree<'a> { let mut id_node: HashMap = HashMap::new(); let mut id_parent: HashMap = HashMap::new(); init_id_parent(root, None, &mut id_node, &mut id_parent); Tree { root, id_node, id_parent, extra_props, } } lazy_static! { static ref APP_NAME_AND_VERSION_RX: regex::Regex = regex::Regex::new("(.+)(-[0-9.]+)").unwrap(); } fn format_marks(marks: &[String]) -> String { if marks.is_empty() { "".to_string() } else { format!("[{}]", marks.join(", ")) } } impl DisplayFormat for DisplayNode<'_> { fn format_for_display(&self, cfg: &config::Config) -> String { let indent = cfg.get_format_indent(); let html_escape = cfg.get_format_html_escape(); let urgency_start = cfg.get_format_urgency_start(); let urgency_end = cfg.get_format_urgency_end(); let icon_dirs = cfg.get_format_icon_dirs(); // fallback_icon has no default value. let fallback_icon: Option> = cfg .get_format_fallback_icon() .as_ref() .map(|i| std::path::Path::new(i).to_owned().into_boxed_path()); let app_name_no_version = APP_NAME_AND_VERSION_RX.replace(self.node.get_app_name(), "$1"); let fmt = match self.node.get_type() { Type::Root => String::from("Cannot format Root"), Type::Output => cfg.get_format_output_format(), Type::Workspace => cfg.get_format_workspace_format(), Type::Container => cfg.get_format_container_format(), Type::Window => cfg.get_format_window_format(), }; let fmt = fmt .replace( "{indent}", indent.repeat(self.get_indent_level()).as_str(), ) .replace( "{urgency_start}", if self.node.urgent { urgency_start.as_str() } else { "" }, ) .replace( "{urgency_end}", if self.node.urgent { urgency_end.as_str() } else { "" }, ) .replace( "{app_icon}", util::get_icon(self.node.get_app_name(), &icon_dirs) .or_else(|| { util::get_icon(&app_name_no_version, &icon_dirs) }) .or_else(|| { util::get_icon( &app_name_no_version.to_lowercase(), &icon_dirs, ) }) .or(fallback_icon) .map(|i| i.to_string_lossy().into_owned()) .unwrap_or_else(String::new) .as_str(), ); fmt_replace!(&fmt, html_escape, { "id" => self.node.id, "app_name" => self.node.get_app_name(), "layout" => format!("{:?}", self.node.layout), "name" | "title" => self.node.get_name(), "output_name" => self .tree .get_parent_node_of_type(self.node.id, Type::Output) .map_or("", |w| w.get_name()), "workspace_name" => self .tree .get_parent_node_of_type(self.node.id, Type::Workspace) .map_or("", |w| w.get_name()), "marks" => format_marks(&self.node.marks), }) } fn get_indent_level(&self) -> usize { match self.indent_level { IndentLevel::Fixed(level) => level as usize, IndentLevel::WorkspacesZeroWindowsOne => { match self.node.get_type(){ Type::Workspace => 0, Type::Window => 1, _ => panic!("Only Workspaces and Windows expected. File a bug report!") } } IndentLevel::TreeDepth(offset) => { let mut depth: usize = 0; let mut node = self.node; while let Some(p) = self.tree.get_parent_node(node.id) { depth += 1; node = p; } if offset > depth { 0 } else { depth - offset as usize } } } } } #[test] fn test_placeholder_rx() { let caps = PLACEHOLDER_RX.captures("Hello, {place}!").unwrap(); assert_eq!(caps.name("name").unwrap().as_str(), "place"); assert_eq!(caps.name("fmtstr"), None); assert_eq!(caps.name("clipstr"), None); let caps = PLACEHOLDER_RX.captures("Hi, {place:{:>10.10}}!").unwrap(); assert_eq!(caps.name("name").unwrap().as_str(), "place"); assert_eq!(caps.name("fmtstr").unwrap().as_str(), "{:>10.10}"); assert_eq!(caps.name("clipstr").unwrap().as_str(), ""); let caps = PLACEHOLDER_RX.captures("Hello, {place:{:.5}…}!").unwrap(); assert_eq!(caps.name("name").unwrap().as_str(), "place"); assert_eq!(caps.name("fmtstr").unwrap().as_str(), "{:.5}"); assert_eq!(caps.name("clipstr").unwrap().as_str(), "…"); let caps = PLACEHOLDER_RX.captures("Hello, {place:{:.5}...}!").unwrap(); assert_eq!(caps.name("name").unwrap().as_str(), "place"); assert_eq!(caps.name("fmtstr").unwrap().as_str(), "{:.5}"); assert_eq!(caps.name("clipstr").unwrap().as_str(), "..."); }