2024-04-16 14:04:34 +02:00
|
|
|
use std::{panic::UnwindSafe, time::Duration};
|
|
|
|
|
2024-04-25 04:37:17 +02:00
|
|
|
use anyhow::anyhow;
|
2024-04-16 14:04:34 +02:00
|
|
|
use pinnacle::{backend::dummy::setup_dummy, state::State};
|
2024-04-16 21:29:53 +02:00
|
|
|
use smithay::{
|
|
|
|
output::Output,
|
|
|
|
reexports::calloop::{
|
|
|
|
self,
|
|
|
|
channel::{Event, Sender},
|
|
|
|
},
|
2024-04-16 14:04:34 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
#[allow(clippy::type_complexity)]
|
|
|
|
pub fn with_state(
|
|
|
|
sender: &Sender<Box<dyn FnOnce(&mut State) + Send>>,
|
|
|
|
with_state: impl FnOnce(&mut State) + Send + 'static,
|
|
|
|
) {
|
|
|
|
sender.send(Box::new(with_state)).unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn sleep_secs(secs: u64) {
|
|
|
|
std::thread::sleep(Duration::from_secs(secs));
|
|
|
|
}
|
|
|
|
|
2024-04-25 04:37:17 +02:00
|
|
|
pub fn test_api<F>(test: F) -> anyhow::Result<()>
|
|
|
|
where
|
|
|
|
F: FnOnce(Sender<Box<dyn FnOnce(&mut State) + Send>>) -> anyhow::Result<()>
|
|
|
|
+ Send
|
|
|
|
+ UnwindSafe
|
|
|
|
+ 'static,
|
|
|
|
{
|
2024-04-16 14:04:34 +02:00
|
|
|
let (mut state, mut event_loop) = setup_dummy(true, None)?;
|
|
|
|
|
|
|
|
let (sender, recv) = calloop::channel::channel::<Box<dyn FnOnce(&mut State) + Send>>();
|
|
|
|
|
|
|
|
event_loop
|
|
|
|
.handle()
|
|
|
|
.insert_source(recv, |event, _, state| match event {
|
|
|
|
Event::Msg(f) => f(state),
|
|
|
|
Event::Closed => (),
|
|
|
|
})
|
|
|
|
.map_err(|_| anyhow::anyhow!("failed to insert source"))?;
|
|
|
|
|
|
|
|
let tempdir = tempfile::tempdir()?;
|
|
|
|
|
|
|
|
state.start_grpc_server(tempdir.path())?;
|
|
|
|
|
|
|
|
let loop_signal = event_loop.get_signal();
|
|
|
|
|
2024-04-25 04:37:17 +02:00
|
|
|
let join_handle = std::thread::spawn(move || -> anyhow::Result<()> {
|
|
|
|
let res = test(sender);
|
2024-04-16 14:04:34 +02:00
|
|
|
loop_signal.stop();
|
2024-04-25 04:37:17 +02:00
|
|
|
res
|
2024-04-16 14:04:34 +02:00
|
|
|
});
|
|
|
|
|
|
|
|
event_loop.run(None, &mut state, |state| {
|
|
|
|
state.fixup_z_layering();
|
|
|
|
state.space.refresh();
|
|
|
|
state.popup_manager.cleanup();
|
|
|
|
|
|
|
|
state
|
|
|
|
.display_handle
|
|
|
|
.flush_clients()
|
|
|
|
.expect("failed to flush client buffers");
|
|
|
|
|
|
|
|
// TODO: couple these or something, this is really error-prone
|
|
|
|
assert_eq!(
|
|
|
|
state.windows.len(),
|
|
|
|
state.z_index_stack.len(),
|
|
|
|
"Length of `windows` and `z_index_stack` are different. \
|
|
|
|
If you see this, report it to the developer."
|
|
|
|
);
|
|
|
|
})?;
|
|
|
|
|
2024-04-25 04:37:17 +02:00
|
|
|
join_handle.join().map_err(|_| anyhow!("thread panicked"))?
|
2024-04-16 14:04:34 +02:00
|
|
|
}
|
2024-04-16 21:29:53 +02:00
|
|
|
|
|
|
|
pub fn output_for_name(state: &State, name: &str) -> Output {
|
|
|
|
state
|
|
|
|
.space
|
|
|
|
.outputs()
|
|
|
|
.find(|op| op.name() == name)
|
|
|
|
.unwrap()
|
|
|
|
.clone()
|
|
|
|
}
|