ewmh.rs 3.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126
  1. extern crate xcb;
  2. use std::mem;
  3. const ATOMS: [&'static str; 3] = [
  4. "_NET_ACTIVE_WINDOW",
  5. "_NET_WM_NAME",
  6. "UTF8_STRING"
  7. ];
  8. #[allow(non_snake_case)]
  9. pub struct EWMH<'a> {
  10. pub conn: &'a xcb::Connection,
  11. pub root: xcb::Window,
  12. pub _NET_ACTIVE_WINDOW: xcb::Atom,
  13. pub _NET_WM_NAME: xcb::Atom,
  14. pub UTF8_STRING: xcb::Atom
  15. }
  16. pub struct WinClass {
  17. pub instance: String,
  18. pub class: String
  19. }
  20. pub fn connect<'a>(conn: &'a xcb::Connection, screen_num: i32) -> EWMH {
  21. let setup = conn.get_setup();
  22. let screen = setup.roots().nth(screen_num as usize).unwrap();
  23. let atoms: Vec<xcb::Atom> = ATOMS
  24. .iter()
  25. .map(|name| xcb::intern_atom(&conn, false, name))
  26. .map(|cookie| cookie.get_reply().unwrap().atom())
  27. .collect();
  28. EWMH {
  29. conn: conn,
  30. root: screen.root(),
  31. _NET_ACTIVE_WINDOW: atoms[0],
  32. _NET_WM_NAME: atoms[1],
  33. UTF8_STRING: atoms[2]
  34. }
  35. }
  36. impl<'a> EWMH<'a> {
  37. pub fn get_active_window(&self) -> xcb::Window {
  38. let cookie = xcb::get_property(&self.conn, false, self.root, self._NET_ACTIVE_WINDOW, xcb::ATOM_WINDOW, 0, 4);
  39. let reply = cookie.get_reply().unwrap();
  40. let value: &xcb::Window = unsafe { mem::transmute(&(reply.value()[0])) };
  41. *value
  42. }
  43. pub fn get_window_name(&self, win: xcb::Window) -> String {
  44. let cookie = xcb::get_property(&self.conn, false, win, self._NET_WM_NAME, self.UTF8_STRING, 0, 100);
  45. let reply = cookie.get_reply();
  46. let value: &str = match reply {
  47. Ok(reply) => unsafe { mem::transmute(reply.value()) },
  48. _ => ""
  49. };
  50. if value == "" {
  51. let cookie = xcb::get_property(&self.conn, false, win, xcb::ATOM_WM_NAME, xcb::ATOM_STRING, 0, 100);
  52. let reply = cookie.get_reply();
  53. let value: &str = match reply {
  54. Ok(reply) => unsafe { mem::transmute(reply.value()) },
  55. _ => ""
  56. };
  57. value.to_string()
  58. }
  59. else {
  60. value.to_string()
  61. }
  62. }
  63. pub fn get_window_class(&self, win: xcb::Window) -> WinClass {
  64. let cookie = xcb::get_property(&self.conn, false, win, xcb::ATOM_WM_CLASS, xcb::ATOM_STRING, 0, 100);
  65. let reply = cookie.get_reply();
  66. let value: &str = match reply {
  67. Ok(reply) => unsafe { mem::transmute(reply.value()) },
  68. _ => ""
  69. };
  70. let parts: Vec<&str> = value.split_terminator('\0').collect();
  71. if parts.len() == 2 {
  72. WinClass {
  73. instance: parts[0].to_string(),
  74. class: parts[1].to_string()
  75. }
  76. }
  77. else {
  78. WinClass {
  79. instance: "".to_string(),
  80. class: "".to_string()
  81. }
  82. }
  83. }
  84. pub fn watch(&self, win: xcb::Window, enabled: bool) {
  85. let value = if enabled {
  86. xcb::EVENT_MASK_PROPERTY_CHANGE
  87. }
  88. else {
  89. xcb::EVENT_MASK_NO_EVENT
  90. };
  91. xcb::change_window_attributes(&self.conn, win, &[(xcb::CW_EVENT_MASK, value)]);
  92. }
  93. pub fn search_by_name(&self, name: &str) -> xcb::Window {
  94. let cookie = xcb::query_tree(&self.conn, self.root);
  95. let reply = cookie.get_reply().unwrap();
  96. for w in reply.children() {
  97. if self.get_window_name(*w) == name {
  98. return *w;
  99. }
  100. }
  101. return 0;
  102. }
  103. pub fn search_by_class(&self, class_name: &str) -> xcb::Window {
  104. let cookie = xcb::query_tree(&self.conn, self.root);
  105. let reply = cookie.get_reply().unwrap();
  106. for w in reply.children() {
  107. if self.get_window_class(*w).class == class_name {
  108. return *w;
  109. }
  110. }
  111. return 0;
  112. }
  113. }