|
|
|
@ -1,7 +1,10 @@
|
|
|
|
|
use std::{
|
|
|
|
|
cell::OnceCell,
|
|
|
|
|
sync::{atomic::AtomicPtr, Arc, Mutex, OnceLock},
|
|
|
|
|
};
|
|
|
|
|
use std::sync::{Arc, Mutex, OnceLock};
|
|
|
|
|
|
|
|
|
|
#[cfg(feature = "crossbeam")]
|
|
|
|
|
use crossbeam_channel::Sender;
|
|
|
|
|
|
|
|
|
|
#[cfg(not(feature = "crossbeam"))]
|
|
|
|
|
use std::sync::mpsc::Sender;
|
|
|
|
|
|
|
|
|
|
use x11rb::{
|
|
|
|
|
connect,
|
|
|
|
@ -65,35 +68,33 @@ fn get_atom<T: Into<String>>(conn: &RustConnection, name: T) -> Option<Atom> {
|
|
|
|
|
|
|
|
|
|
//Not very pretty and a bit wasteful making a new connection and getting the atom every time..
|
|
|
|
|
pub fn focus_target() {
|
|
|
|
|
unsafe {
|
|
|
|
|
if let Some(target) = TARGET.get() {
|
|
|
|
|
if let Ok(target) = target.lock() {
|
|
|
|
|
if *target != x11rb::NONE {
|
|
|
|
|
if let Ok((conn, _screen_num)) = connect(None) {
|
|
|
|
|
let net_active_window = get_atom(&conn, "_NET_ACTIVE_WINDOW").unwrap();
|
|
|
|
|
|
|
|
|
|
let event = ClientMessageEvent {
|
|
|
|
|
response_type: xproto::CLIENT_MESSAGE_EVENT,
|
|
|
|
|
format: 32,
|
|
|
|
|
sequence: 0,
|
|
|
|
|
window: *target,
|
|
|
|
|
type_: net_active_window,
|
|
|
|
|
data: ClientMessageData::from([1, x11rb::CURRENT_TIME, 0, 0, 0]),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if let Ok(cookie) = conn.send_event(
|
|
|
|
|
false,
|
|
|
|
|
conn.setup().roots[_screen_num].root,
|
|
|
|
|
EventMask::SUBSTRUCTURE_NOTIFY | EventMask::SUBSTRUCTURE_REDIRECT,
|
|
|
|
|
event,
|
|
|
|
|
) {
|
|
|
|
|
if let Err(e) = cookie.check() {
|
|
|
|
|
log::error!("Error focusing target: {e:?}");
|
|
|
|
|
}
|
|
|
|
|
if let Some(target) = TARGET.get() {
|
|
|
|
|
if let Ok(target) = target.lock() {
|
|
|
|
|
if *target != x11rb::NONE {
|
|
|
|
|
if let Ok((conn, _screen_num)) = connect(None) {
|
|
|
|
|
let net_active_window = get_atom(&conn, "_NET_ACTIVE_WINDOW").unwrap();
|
|
|
|
|
|
|
|
|
|
let event = ClientMessageEvent {
|
|
|
|
|
response_type: xproto::CLIENT_MESSAGE_EVENT,
|
|
|
|
|
format: 32,
|
|
|
|
|
sequence: 0,
|
|
|
|
|
window: *target,
|
|
|
|
|
type_: net_active_window,
|
|
|
|
|
data: ClientMessageData::from([1, x11rb::CURRENT_TIME, 0, 0, 0]),
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
if let Ok(cookie) = conn.send_event(
|
|
|
|
|
false,
|
|
|
|
|
conn.setup().roots[_screen_num].root,
|
|
|
|
|
EventMask::SUBSTRUCTURE_NOTIFY | EventMask::SUBSTRUCTURE_REDIRECT,
|
|
|
|
|
event,
|
|
|
|
|
) {
|
|
|
|
|
if let Err(e) = cookie.check() {
|
|
|
|
|
log::error!("Error focusing target: {e:?}");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
conn.flush().ok();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
conn.flush().ok();
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
@ -105,7 +106,7 @@ struct Runtime {
|
|
|
|
|
root: Window,
|
|
|
|
|
target: Target,
|
|
|
|
|
atoms: AtomCollection,
|
|
|
|
|
underlay_tx: std::sync::mpsc::Sender<UnderlayEvent>,
|
|
|
|
|
underlay_tx: Sender<UnderlayEvent>,
|
|
|
|
|
active: Window,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
@ -133,11 +134,9 @@ impl Runtime {
|
|
|
|
|
///Using this instead to allow setting the globally accessible target too for focus_target.
|
|
|
|
|
fn set_target(&mut self, wid: Window) {
|
|
|
|
|
self.target.wid = wid;
|
|
|
|
|
unsafe {
|
|
|
|
|
if let Some(target) = TARGET.get() {
|
|
|
|
|
if let Ok(mut target) = target.lock() {
|
|
|
|
|
*target = wid;
|
|
|
|
|
}
|
|
|
|
|
if let Some(target) = TARGET.get() {
|
|
|
|
|
if let Ok(mut target) = target.lock() {
|
|
|
|
|
*target = wid;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
@ -360,10 +359,7 @@ impl Runtime {
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn init(
|
|
|
|
|
window_title: String,
|
|
|
|
|
tx: std::sync::mpsc::Sender<crate::UnderlayEvent>,
|
|
|
|
|
) -> Result<(), String> {
|
|
|
|
|
pub fn init(window_title: String, tx: Sender<crate::UnderlayEvent>) -> Result<(), String> {
|
|
|
|
|
TARGET
|
|
|
|
|
.set(Arc::new(Mutex::new(Window::default())))
|
|
|
|
|
.map_err(|e| log::error!("Could not init global TARGET storage {e:?}"))
|
|
|
|
|