panel.rs 2.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192
  1. use xcb;
  2. use ui::tray;
  3. use ui::x11;
  4. use std::cell::RefCell;
  5. use std::rc::Rc;
  6. pub struct Panel {
  7. pub conn: x11::Connection,
  8. pub window: xcb::Window,
  9. pub width: u16,
  10. tray: RefCell<Option<tray::Tray>>,
  11. finishing: RefCell<bool>
  12. }
  13. impl Panel {
  14. pub fn new(conn: x11::Connection) -> Rc<Panel> {
  15. let panel = Rc::new(Panel {
  16. window: conn.generate_id(),
  17. width: conn.default_screen().width,
  18. conn: conn,
  19. tray: RefCell::new(None),
  20. finishing: RefCell::new(false)
  21. });
  22. let tray = tray::Tray::new(Rc::downgrade(&panel));
  23. *panel.tray.borrow_mut() = Some(tray);
  24. panel
  25. }
  26. pub fn create(&self) {
  27. let conn = &self.conn;
  28. let screen = conn.default_screen();
  29. xcb::create_window(
  30. conn,
  31. xcb::COPY_FROM_PARENT as u8,
  32. self.window,
  33. screen.root,
  34. 0, 0,
  35. screen.width, 20,
  36. 0,
  37. xcb::WINDOW_CLASS_INPUT_OUTPUT as u16,
  38. xcb::COPY_FROM_PARENT,
  39. &[
  40. (xcb::CW_BACK_PIXEL, 0xFF000000),
  41. (xcb::CW_EVENT_MASK, xcb::EVENT_MASK_PROPERTY_CHANGE | xcb::EVENT_MASK_STRUCTURE_NOTIFY)
  42. ]
  43. );
  44. self.set_property(
  45. conn.atom(x11::_NET_WM_WINDOW_TYPE),
  46. xcb::ATOM_ATOM,
  47. 32,
  48. &[conn.atom(x11::_NET_WM_WINDOW_TYPE_DOCK)]
  49. );
  50. xcb::map_window(conn, self.window);
  51. conn.flush();
  52. }
  53. pub fn set_property<T>(&self, name: xcb::Atom, type_: xcb::Atom, format: u8, data: &[T]) {
  54. xcb::change_property(
  55. &self.conn,
  56. xcb::PROP_MODE_REPLACE as u8,
  57. self.window,
  58. name,
  59. type_,
  60. format,
  61. data
  62. );
  63. }
  64. pub fn handle_event(&self, event: xcb::GenericEvent) -> bool {
  65. let tray = self.tray.borrow_mut().as_mut().unwrap().handle_event(&event);
  66. if self.is_finishing() && event.response_type() == xcb::DESTROY_NOTIFY {
  67. let event: &xcb::DestroyNotifyEvent = xcb::cast_event(&event);
  68. if event.window() == self.window {
  69. return true
  70. }
  71. }
  72. tray
  73. }
  74. pub fn is_finishing(&self) -> bool {
  75. *self.finishing.borrow()
  76. }
  77. pub fn finish(&self) {
  78. *self.finishing.borrow_mut() = true;
  79. self.tray.borrow_mut().as_mut().unwrap().finish();
  80. xcb::destroy_window(&self.conn, self.window);
  81. self.conn.flush();
  82. }
  83. }