2024-03-07 01:50:45 +01:00
|
|
|
use std::{
|
|
|
|
io::Write,
|
2024-03-07 05:56:22 +01:00
|
|
|
panic::UnwindSafe,
|
2024-03-07 01:50:45 +01:00
|
|
|
process::{Command, Stdio},
|
|
|
|
time::Duration,
|
|
|
|
};
|
|
|
|
|
2024-03-07 05:56:22 +01:00
|
|
|
use pinnacle::{
|
|
|
|
backend::dummy::setup_dummy,
|
|
|
|
state::{State, WithState},
|
|
|
|
};
|
2024-03-07 01:50:45 +01:00
|
|
|
use smithay::reexports::calloop::{
|
|
|
|
self,
|
|
|
|
channel::{Event, Sender},
|
|
|
|
};
|
|
|
|
|
|
|
|
use test_log::test;
|
|
|
|
|
|
|
|
fn run_lua(ident: &str, code: &str) {
|
2024-03-07 05:56:22 +01:00
|
|
|
#[rustfmt::skip]
|
|
|
|
let code = format!(r#"
|
2024-03-23 07:39:07 +01:00
|
|
|
require("pinnacle").run(function({ident})
|
2024-03-07 05:56:22 +01:00
|
|
|
local run = function({ident})
|
|
|
|
{code}
|
|
|
|
end
|
|
|
|
|
|
|
|
local success, err = pcall(run, {ident})
|
|
|
|
|
|
|
|
if not success then
|
|
|
|
print(err)
|
|
|
|
os.exit(1)
|
|
|
|
end
|
|
|
|
end)
|
|
|
|
"#);
|
2024-03-07 01:50:45 +01:00
|
|
|
|
|
|
|
let mut child = Command::new("lua").stdin(Stdio::piped()).spawn().unwrap();
|
|
|
|
|
2024-03-07 05:56:22 +01:00
|
|
|
let mut stdin = child.stdin.take().unwrap();
|
2024-03-07 01:50:45 +01:00
|
|
|
|
|
|
|
stdin.write_all(code.as_bytes()).unwrap();
|
|
|
|
|
|
|
|
drop(stdin);
|
|
|
|
|
2024-03-07 05:56:22 +01:00
|
|
|
let exit_status = child.wait().unwrap();
|
|
|
|
|
|
|
|
if exit_status.code().is_some_and(|code| code != 0) {
|
|
|
|
panic!("lua code panicked");
|
|
|
|
}
|
2024-03-07 01:50:45 +01:00
|
|
|
}
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
struct SetupLuaGuard {
|
|
|
|
child: std::process::Child,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for SetupLuaGuard {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
let _ = self.child.kill();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[must_use]
|
|
|
|
fn setup_lua(ident: &str, code: &str) -> SetupLuaGuard {
|
|
|
|
#[rustfmt::skip]
|
|
|
|
let code = format!(r#"
|
|
|
|
require("pinnacle").setup(function({ident})
|
|
|
|
local run = function({ident})
|
|
|
|
{code}
|
|
|
|
end
|
|
|
|
|
|
|
|
local success, err = pcall(run, {ident})
|
|
|
|
|
|
|
|
if not success then
|
|
|
|
print(err)
|
|
|
|
os.exit(1)
|
|
|
|
end
|
|
|
|
end)
|
|
|
|
"#);
|
|
|
|
|
|
|
|
let mut child = Command::new("lua").stdin(Stdio::piped()).spawn().unwrap();
|
|
|
|
|
|
|
|
let mut stdin = child.stdin.take().unwrap();
|
|
|
|
|
|
|
|
stdin.write_all(code.as_bytes()).unwrap();
|
|
|
|
|
|
|
|
drop(stdin);
|
|
|
|
|
|
|
|
SetupLuaGuard { child }
|
|
|
|
|
|
|
|
// let exit_status = child.wait().unwrap();
|
|
|
|
//
|
|
|
|
// if exit_status.code().is_some_and(|code| code != 0) {
|
|
|
|
// panic!("lua code panicked");
|
|
|
|
// }
|
|
|
|
}
|
|
|
|
|
2024-03-07 01:50:45 +01:00
|
|
|
#[allow(clippy::type_complexity)]
|
2024-04-15 22:50:09 +02:00
|
|
|
fn with_state(
|
2024-03-07 01:50:45 +01:00
|
|
|
sender: &Sender<Box<dyn FnOnce(&mut State) + Send>>,
|
|
|
|
assert: impl FnOnce(&mut State) + Send + 'static,
|
|
|
|
) {
|
|
|
|
sender.send(Box::new(assert)).unwrap();
|
|
|
|
}
|
|
|
|
|
|
|
|
fn sleep_secs(secs: u64) {
|
|
|
|
std::thread::sleep(Duration::from_secs(secs));
|
|
|
|
}
|
|
|
|
|
|
|
|
macro_rules! run_lua {
|
|
|
|
{ |$ident:ident| $($body:tt)* } => {
|
|
|
|
run_lua(stringify!($ident), stringify!($($body)*));
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
macro_rules! setup_lua {
|
|
|
|
{ |$ident:ident| $($body:tt)* } => {
|
|
|
|
let _guard = setup_lua(stringify!($ident), stringify!($($body)*));
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2024-03-07 01:50:45 +01:00
|
|
|
fn test_lua_api(
|
2024-03-07 05:56:22 +01:00
|
|
|
test: impl FnOnce(Sender<Box<dyn FnOnce(&mut State) + Send>>) + Send + UnwindSafe + 'static,
|
2024-03-07 01:50:45 +01:00
|
|
|
) -> anyhow::Result<()> {
|
|
|
|
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),
|
2024-03-07 05:56:22 +01:00
|
|
|
Event::Closed => (),
|
2024-03-07 01:50:45 +01:00
|
|
|
})
|
|
|
|
.map_err(|_| anyhow::anyhow!("failed to insert source"))?;
|
|
|
|
|
|
|
|
let tempdir = tempfile::tempdir()?;
|
|
|
|
|
|
|
|
state.start_grpc_server(tempdir.path())?;
|
|
|
|
|
2024-03-07 05:56:22 +01:00
|
|
|
let loop_signal = event_loop.get_signal();
|
|
|
|
|
|
|
|
let join_handle = std::thread::spawn(move || {
|
|
|
|
let res = std::panic::catch_unwind(|| {
|
|
|
|
test(sender);
|
|
|
|
});
|
|
|
|
loop_signal.stop();
|
|
|
|
if let Err(err) = res {
|
|
|
|
std::panic::resume_unwind(err);
|
|
|
|
}
|
|
|
|
});
|
2024-03-07 01:50:45 +01: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-03-07 05:56:22 +01:00
|
|
|
if let Err(err) = join_handle.join() {
|
|
|
|
panic!("{err:?}");
|
|
|
|
}
|
|
|
|
|
2024-03-07 01:50:45 +01:00
|
|
|
Ok(())
|
|
|
|
}
|
|
|
|
|
2024-03-07 05:56:22 +01:00
|
|
|
mod coverage {
|
|
|
|
use pinnacle::{
|
|
|
|
tag::TagId,
|
|
|
|
window::{
|
|
|
|
rules::{WindowRule, WindowRuleCondition},
|
|
|
|
window_state::FullscreenOrMaximized,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
// Process
|
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
mod process {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[tokio::main]
|
|
|
|
#[self::test]
|
|
|
|
async fn spawn() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|sender| {
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
sleep_secs(1);
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| {
|
2024-04-11 19:35:48 +02:00
|
|
|
assert_eq!(state.windows.len(), 1);
|
|
|
|
assert_eq!(state.windows[0].class(), Some("foot".to_string()));
|
|
|
|
});
|
|
|
|
})
|
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
#[tokio::main]
|
|
|
|
#[self::test]
|
|
|
|
async fn set_env() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|sender| {
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.process.set_env("PROCESS_SET_ENV", "env value")
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |_state| {
|
2024-04-11 19:35:48 +02:00
|
|
|
assert_eq!(
|
|
|
|
std::env::var("PROCESS_SET_ENV"),
|
|
|
|
Ok("env value".to_string())
|
|
|
|
);
|
|
|
|
});
|
|
|
|
})
|
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
// Window
|
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
mod window {
|
|
|
|
use super::*;
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
#[tokio::main]
|
|
|
|
#[self::test]
|
|
|
|
async fn get_all() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|_sender| {
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
assert(#Pinnacle.window.get_all() == 0)
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
for i = 1, 5 do
|
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
end
|
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
sleep_secs(1);
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
assert(#Pinnacle.window.get_all() == 5)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
#[tokio::main]
|
|
|
|
#[self::test]
|
|
|
|
async fn get_focused() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|_sender| {
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
assert(not Pinnacle.window.get_focused())
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
Pinnacle.tag.add(Pinnacle.output.get_focused(), "1")[1]:set_active(true)
|
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
sleep_secs(1);
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
assert(Pinnacle.window.get_focused())
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
#[tokio::main]
|
|
|
|
#[self::test]
|
|
|
|
async fn add_window_rule() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|sender| {
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.tag.add(Pinnacle.output.get_focused(), "Tag Name")
|
|
|
|
Pinnacle.window.add_window_rule({
|
|
|
|
cond = { classes = { "firefox" } },
|
|
|
|
rule = { tags = { Pinnacle.tag.get("Tag Name") } },
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| {
|
2024-04-11 19:35:48 +02:00
|
|
|
assert_eq!(state.config.window_rules.len(), 1);
|
|
|
|
assert_eq!(
|
|
|
|
state.config.window_rules[0],
|
|
|
|
(
|
|
|
|
WindowRuleCondition {
|
|
|
|
class: Some(vec!["firefox".to_string()]),
|
|
|
|
..Default::default()
|
|
|
|
},
|
|
|
|
WindowRule {
|
|
|
|
tags: Some(vec![TagId(0)]),
|
2024-03-07 05:56:22 +01:00
|
|
|
..Default::default()
|
2024-04-11 19:35:48 +02:00
|
|
|
}
|
|
|
|
)
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.tag.add(Pinnacle.output.get_focused(), "Tag Name 2")
|
|
|
|
Pinnacle.window.add_window_rule({
|
|
|
|
cond = {
|
|
|
|
all = {
|
|
|
|
{
|
|
|
|
classes = { "steam" },
|
|
|
|
tags = {
|
|
|
|
Pinnacle.tag.get("Tag Name"),
|
|
|
|
Pinnacle.tag.get("Tag Name 2"),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
},
|
2024-04-11 19:35:48 +02:00
|
|
|
rule = { fullscreen_or_maximized = "fullscreen" },
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| {
|
2024-04-11 19:35:48 +02:00
|
|
|
assert_eq!(state.config.window_rules.len(), 2);
|
|
|
|
assert_eq!(
|
|
|
|
state.config.window_rules[1],
|
|
|
|
(
|
|
|
|
WindowRuleCondition {
|
|
|
|
cond_all: Some(vec![WindowRuleCondition {
|
|
|
|
class: Some(vec!["steam".to_string()]),
|
|
|
|
tag: Some(vec![TagId(0), TagId(1)]),
|
|
|
|
..Default::default()
|
|
|
|
}]),
|
|
|
|
..Default::default()
|
|
|
|
},
|
|
|
|
WindowRule {
|
|
|
|
fullscreen_or_maximized: Some(FullscreenOrMaximized::Fullscreen),
|
|
|
|
..Default::default()
|
|
|
|
}
|
|
|
|
)
|
|
|
|
);
|
|
|
|
});
|
|
|
|
})
|
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
// TODO: window_begin_move
|
|
|
|
// TODO: window_begin_resize
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
mod handle {
|
|
|
|
use super::*;
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
// WindowHandle
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
#[tokio::main]
|
|
|
|
#[self::test]
|
|
|
|
async fn close() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|sender| {
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
sleep_secs(1);
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| {
|
2024-04-11 19:35:48 +02:00
|
|
|
assert_eq!(state.windows.len(), 1);
|
|
|
|
});
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.window.get_all()[1]:close()
|
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
sleep_secs(1);
|
2024-03-07 05:56:22 +01:00
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| {
|
2024-04-11 19:35:48 +02:00
|
|
|
assert_eq!(state.windows.len(), 0);
|
|
|
|
});
|
|
|
|
})
|
2024-03-07 05:56:22 +01:00
|
|
|
}
|
|
|
|
|
2024-04-11 19:35:48 +02:00
|
|
|
#[tokio::main]
|
|
|
|
#[self::test]
|
|
|
|
async fn move_to_tag() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|sender| {
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
local tags = Pinnacle.tag.add(Pinnacle.output.get_focused(), "1", "2", "3")
|
|
|
|
tags[1]:set_active(true)
|
|
|
|
tags[2]:set_active(true)
|
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| {
|
2024-04-11 19:35:48 +02:00
|
|
|
assert_eq!(
|
|
|
|
state.windows[0].with_state(|st| st
|
|
|
|
.tags
|
|
|
|
.iter()
|
|
|
|
.map(|tag| tag.name())
|
|
|
|
.collect::<Vec<_>>()),
|
|
|
|
vec!["1", "2"]
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Correct usage
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.window.get_all()[1]:move_to_tag(Pinnacle.tag.get("3"))
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| {
|
2024-04-11 19:35:48 +02:00
|
|
|
assert_eq!(
|
|
|
|
state.windows[0].with_state(|st| st
|
|
|
|
.tags
|
|
|
|
.iter()
|
|
|
|
.map(|tag| tag.name())
|
|
|
|
.collect::<Vec<_>>()),
|
|
|
|
vec!["3"]
|
|
|
|
);
|
|
|
|
});
|
|
|
|
|
|
|
|
// Move to the same tag
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.window.get_all()[1]:move_to_tag(Pinnacle.tag.get("3"))
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| {
|
2024-04-11 19:35:48 +02:00
|
|
|
assert_eq!(
|
|
|
|
state.windows[0].with_state(|st| st
|
|
|
|
.tags
|
|
|
|
.iter()
|
|
|
|
.map(|tag| tag.name())
|
|
|
|
.collect::<Vec<_>>()),
|
|
|
|
vec!["3"]
|
|
|
|
);
|
|
|
|
});
|
|
|
|
})
|
2024-03-07 05:56:22 +01:00
|
|
|
}
|
2024-04-11 19:35:48 +02:00
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
}
|
2024-04-12 14:57:22 +02:00
|
|
|
|
|
|
|
mod tag {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
mod handle {
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[tokio::main]
|
|
|
|
#[self::test]
|
|
|
|
async fn props() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|_sender| {
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.output.connect_for_all(function(op)
|
|
|
|
local tags = Pinnacle.tag.add(op, "First", "Mungus", "Potato")
|
|
|
|
tags[1]:set_active(true)
|
|
|
|
tags[3]:set_active(true)
|
|
|
|
end)
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
local first_props = Pinnacle.tag.get("First"):props()
|
|
|
|
assert(first_props.active == true)
|
|
|
|
assert(first_props.name == "First")
|
|
|
|
assert(first_props.output.name == "Pinnacle Window")
|
|
|
|
assert(#first_props.windows == 2)
|
|
|
|
assert(first_props.windows[1]:class() == "foot")
|
|
|
|
assert(first_props.windows[2]:class() == "foot")
|
|
|
|
|
|
|
|
local mungus_props = Pinnacle.tag.get("Mungus"):props()
|
|
|
|
assert(mungus_props.active == false)
|
|
|
|
assert(mungus_props.name == "Mungus")
|
|
|
|
assert(mungus_props.output.name == "Pinnacle Window")
|
|
|
|
assert(#mungus_props.windows == 0)
|
|
|
|
|
|
|
|
local potato_props = Pinnacle.tag.get("Potato"):props()
|
|
|
|
assert(potato_props.active == true)
|
|
|
|
assert(potato_props.name == "Potato")
|
|
|
|
assert(potato_props.output.name == "Pinnacle Window")
|
|
|
|
assert(#potato_props.windows == 2)
|
|
|
|
assert(first_props.windows[1]:class() == "foot")
|
|
|
|
assert(first_props.windows[2]:class() == "foot")
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2024-04-15 22:50:09 +02:00
|
|
|
|
|
|
|
mod output {
|
|
|
|
use smithay::utils::Rectangle;
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[tokio::main]
|
|
|
|
#[self::test]
|
|
|
|
async fn setup() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|sender| {
|
|
|
|
setup_lua! { |Pinnacle|
|
|
|
|
Pinnacle.output.setup({
|
|
|
|
{
|
|
|
|
function(_)
|
|
|
|
return true
|
|
|
|
end,
|
|
|
|
tag_names = { "First", "Third", "Schmurd" },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"Pinnacle Window",
|
|
|
|
loc = { x = 300, y = 0 },
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"Output 1",
|
|
|
|
loc = { "Pinnacle Window", "bottom_align_left" },
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
|
|
|
with_state(&sender, |state| {
|
|
|
|
state.new_output("Output 1", (960, 540).into());
|
|
|
|
});
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
|
|
|
with_state(&sender, |state| {
|
|
|
|
let original_op = state
|
|
|
|
.space
|
|
|
|
.outputs()
|
|
|
|
.find(|op| op.name() == "Pinnacle Window")
|
|
|
|
.unwrap();
|
|
|
|
let output_1 = state
|
|
|
|
.space
|
|
|
|
.outputs()
|
|
|
|
.find(|op| op.name() == "Output 1")
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let original_op_geo = state.space.output_geometry(original_op).unwrap();
|
|
|
|
let output_1_geo = state.space.output_geometry(output_1).unwrap();
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
original_op_geo,
|
|
|
|
Rectangle::from_loc_and_size((300, 0), (1920, 1080))
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
output_1_geo,
|
|
|
|
Rectangle::from_loc_and_size((300, 1080), (960, 540))
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
output_1.with_state(|state| state
|
|
|
|
.tags
|
|
|
|
.iter()
|
|
|
|
.map(|tag| tag.name())
|
|
|
|
.collect::<Vec<_>>()),
|
|
|
|
vec!["First", "Third", "Schmurd"]
|
|
|
|
);
|
|
|
|
|
|
|
|
state.remove_output(&original_op.clone());
|
|
|
|
});
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
|
|
|
with_state(&sender, |state| {
|
|
|
|
let output_1 = state
|
|
|
|
.space
|
|
|
|
.outputs()
|
|
|
|
.find(|op| op.name() == "Output 1")
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let output_1_geo = state.space.output_geometry(output_1).unwrap();
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
output_1_geo,
|
|
|
|
Rectangle::from_loc_and_size((0, 0), (960, 540))
|
|
|
|
);
|
|
|
|
|
|
|
|
state.new_output("Output 2", (300, 500).into());
|
|
|
|
});
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
|
|
|
with_state(&sender, |state| {
|
|
|
|
let output_1 = state
|
|
|
|
.space
|
|
|
|
.outputs()
|
|
|
|
.find(|op| op.name() == "Output 1")
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let output_2 = state
|
|
|
|
.space
|
|
|
|
.outputs()
|
|
|
|
.find(|op| op.name() == "Output 2")
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
let output_1_geo = state.space.output_geometry(output_1).unwrap();
|
|
|
|
let output_2_geo = state.space.output_geometry(output_2).unwrap();
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
output_2_geo,
|
|
|
|
Rectangle::from_loc_and_size((0, 0), (300, 500))
|
|
|
|
);
|
|
|
|
|
|
|
|
assert_eq!(
|
|
|
|
output_1_geo,
|
|
|
|
Rectangle::from_loc_and_size((300, 0), (960, 540))
|
|
|
|
);
|
|
|
|
});
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
}
|
|
|
|
|
2024-03-07 01:50:45 +01:00
|
|
|
#[tokio::main]
|
|
|
|
#[test]
|
|
|
|
async fn window_count_with_tag_is_correct() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|sender| {
|
|
|
|
run_lua! { |Pinnacle|
|
2024-03-07 05:56:22 +01:00
|
|
|
Pinnacle.tag.add(Pinnacle.output.get_focused(), "1")
|
2024-03-07 01:50:45 +01:00
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| assert_eq!(state.windows.len(), 1));
|
2024-03-07 01:50:45 +01:00
|
|
|
|
2024-03-07 05:56:22 +01:00
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
for i = 1, 20 do
|
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
end
|
|
|
|
}
|
|
|
|
|
2024-03-07 01:50:45 +01:00
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| assert_eq!(state.windows.len(), 21));
|
2024-03-07 05:56:22 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::main]
|
|
|
|
#[test]
|
|
|
|
async fn window_count_without_tag_is_correct() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|sender| {
|
2024-03-07 01:50:45 +01:00
|
|
|
run_lua! { |Pinnacle|
|
2024-03-07 05:56:22 +01:00
|
|
|
Pinnacle.process.spawn("foot")
|
2024-03-07 01:50:45 +01:00
|
|
|
}
|
2024-03-07 05:56:22 +01:00
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| assert_eq!(state.windows.len(), 1));
|
2024-03-07 05:56:22 +01:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::main]
|
|
|
|
#[test]
|
|
|
|
async fn spawned_window_on_active_tag_has_keyboard_focus() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|sender| {
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.tag.add(Pinnacle.output.get_focused(), "1")[1]:set_active(true)
|
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| {
|
2024-03-07 05:56:22 +01:00
|
|
|
assert_eq!(
|
|
|
|
state
|
|
|
|
.focused_window(state.focused_output().unwrap())
|
|
|
|
.unwrap()
|
|
|
|
.class(),
|
|
|
|
Some("foot".to_string())
|
|
|
|
);
|
|
|
|
});
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::main]
|
|
|
|
#[test]
|
|
|
|
async fn spawned_window_on_inactive_tag_does_not_have_keyboard_focus() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|sender| {
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.tag.add(Pinnacle.output.get_focused(), "1")
|
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| {
|
2024-03-07 05:56:22 +01:00
|
|
|
assert_eq!(state.focused_window(state.focused_output().unwrap()), None);
|
|
|
|
});
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[tokio::main]
|
|
|
|
#[test]
|
|
|
|
async fn spawned_window_has_correct_tags() -> anyhow::Result<()> {
|
|
|
|
test_lua_api(|sender| {
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.tag.add(Pinnacle.output.get_focused(), "1", "2", "3")
|
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| {
|
2024-03-07 05:56:22 +01:00
|
|
|
assert_eq!(state.windows.len(), 1);
|
|
|
|
assert_eq!(state.windows[0].with_state(|st| st.tags.len()), 1);
|
|
|
|
});
|
|
|
|
|
|
|
|
run_lua! { |Pinnacle|
|
|
|
|
Pinnacle.tag.get("1"):set_active(true)
|
|
|
|
Pinnacle.tag.get("3"):set_active(true)
|
|
|
|
Pinnacle.process.spawn("foot")
|
|
|
|
}
|
|
|
|
|
|
|
|
sleep_secs(1);
|
|
|
|
|
2024-04-15 22:50:09 +02:00
|
|
|
with_state(&sender, |state| {
|
2024-03-07 05:56:22 +01:00
|
|
|
assert_eq!(state.windows.len(), 2);
|
|
|
|
assert_eq!(state.windows[1].with_state(|st| st.tags.len()), 2);
|
|
|
|
assert_eq!(
|
|
|
|
state.windows[1].with_state(|st| st
|
|
|
|
.tags
|
|
|
|
.iter()
|
|
|
|
.map(|tag| tag.name())
|
|
|
|
.collect::<Vec<_>>()),
|
|
|
|
vec!["1", "3"]
|
|
|
|
);
|
|
|
|
});
|
2024-03-07 01:50:45 +01:00
|
|
|
})
|
|
|
|
}
|