2025-10-29 23:21:28 +00:00
|
|
|
#include "../shared.hpp"
|
|
|
|
|
#include "../../shared.hpp"
|
|
|
|
|
#include "../../hyprctlCompat.hpp"
|
|
|
|
|
#include "tests.hpp"
|
|
|
|
|
|
|
|
|
|
static int ret = 0;
|
|
|
|
|
|
|
|
|
|
static void testFloatClamp() {
|
|
|
|
|
for (auto const& win : {"a", "b", "c"}) {
|
|
|
|
|
if (!Tests::spawnKitty(win)) {
|
|
|
|
|
NLog::log("{}Failed to spawn kitty with win class `{}`", Colors::RED, win);
|
|
|
|
|
++TESTS_FAILED;
|
|
|
|
|
ret = 1;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/keyword dwindle:force_split 2"));
|
2025-11-13 00:08:04 +00:00
|
|
|
OK(getFromSocket("/keyword monitor HEADLESS-2, addreserved, 0, 20, 0, 20"));
|
2025-10-29 23:21:28 +00:00
|
|
|
OK(getFromSocket("/dispatch focuswindow class:c"));
|
|
|
|
|
OK(getFromSocket("/dispatch setfloating class:c"));
|
|
|
|
|
OK(getFromSocket("/dispatch resizewindowpixel exact 1200 900,class:c"));
|
|
|
|
|
OK(getFromSocket("/dispatch settiled class:c"));
|
|
|
|
|
OK(getFromSocket("/dispatch setfloating class:c"));
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/clients");
|
2025-11-13 00:08:04 +00:00
|
|
|
EXPECT_CONTAINS(str, "at: 698,158");
|
2025-10-29 23:21:28 +00:00
|
|
|
EXPECT_CONTAINS(str, "size: 1200,900");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/keyword dwindle:force_split 0"));
|
|
|
|
|
|
|
|
|
|
// clean up
|
|
|
|
|
NLog::log("{}Killing all windows", Colors::YELLOW);
|
|
|
|
|
Tests::killAllWindows();
|
2026-02-26 12:00:05 +00:00
|
|
|
|
|
|
|
|
OK(getFromSocket("/reload"));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void test13349() {
|
|
|
|
|
|
|
|
|
|
// Test if dwindle properly uses a focal point to place a new window.
|
|
|
|
|
// exposed by #13349 as a regression from #12890
|
|
|
|
|
|
|
|
|
|
for (auto const& win : {"a", "b", "c"}) {
|
|
|
|
|
if (!Tests::spawnKitty(win)) {
|
|
|
|
|
NLog::log("{}Failed to spawn kitty with win class `{}`", Colors::RED, win);
|
|
|
|
|
++TESTS_FAILED;
|
|
|
|
|
ret = 1;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch focuswindow class:c"));
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/activewindow");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 967,547");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 931,511");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch movewindow l"));
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/activewindow");
|
2026-03-02 21:12:27 +00:00
|
|
|
EXPECT_CONTAINS(str, "at: 497,22");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 456,1036");
|
2026-02-26 12:00:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch movewindow r"));
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/activewindow");
|
2026-03-02 21:12:27 +00:00
|
|
|
EXPECT_CONTAINS(str, "at: 967,22");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 456,1036");
|
2026-02-26 12:00:05 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// clean up
|
|
|
|
|
NLog::log("{}Killing all windows", Colors::YELLOW);
|
|
|
|
|
Tests::killAllWindows();
|
2025-10-29 23:21:28 +00:00
|
|
|
}
|
|
|
|
|
|
2026-03-02 16:21:20 +00:00
|
|
|
static void testSplit() {
|
|
|
|
|
// Test various split methods
|
|
|
|
|
|
2026-03-03 11:55:57 +00:00
|
|
|
Tests::spawnKitty("a");
|
|
|
|
|
|
|
|
|
|
// these must not crash
|
|
|
|
|
EXPECT_NOT(getFromSocket("/dispatch layoutmsg swapsplit"), "ok");
|
|
|
|
|
EXPECT_NOT(getFromSocket("/dispatch layoutmsg splitratio 1 exact"), "ok");
|
|
|
|
|
|
|
|
|
|
Tests::spawnKitty("b");
|
2026-03-02 16:21:20 +00:00
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch focuswindow class:a"));
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg splitratio -0.2"));
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/activewindow");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 22,22");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 743,1036");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg splitratio 1.6 exact"));
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/activewindow");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 22,22");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 1495,1036");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
EXPECT_NOT(getFromSocket("/dispatch layoutmsg splitratio fhne exact"), "ok");
|
|
|
|
|
EXPECT_NOT(getFromSocket("/dispatch layoutmsg splitratio exact"), "ok");
|
|
|
|
|
EXPECT_NOT(getFromSocket("/dispatch layoutmsg splitratio -....9"), "ok");
|
|
|
|
|
EXPECT_NOT(getFromSocket("/dispatch layoutmsg splitratio ..9"), "ok");
|
|
|
|
|
EXPECT_NOT(getFromSocket("/dispatch layoutmsg splitratio"), "ok");
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg togglesplit"));
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/activewindow");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 22,22");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 1876,823");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg swapsplit"));
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/activewindow");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 22,859");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 1876,199");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NLog::log("{}Killing all windows", Colors::YELLOW);
|
|
|
|
|
Tests::killAllWindows();
|
|
|
|
|
}
|
|
|
|
|
|
2026-03-06 23:47:48 +03:00
|
|
|
static void testRotatesplit() {
|
|
|
|
|
OK(getFromSocket("r/keyword general:gaps_in 0"));
|
|
|
|
|
OK(getFromSocket("r/keyword general:gaps_out 0"));
|
|
|
|
|
OK(getFromSocket("r/keyword general:border_size 0"));
|
|
|
|
|
|
|
|
|
|
for (auto const& win : {"a", "b"}) {
|
|
|
|
|
if (!Tests::spawnKitty(win)) {
|
|
|
|
|
NLog::log("{}Failed to spawn kitty with win class `{}`", Colors::RED, win);
|
|
|
|
|
++TESTS_FAILED;
|
|
|
|
|
ret = 1;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/clients");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 0,0");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 960,1080");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// test 4 repeated rotations by 90 degrees
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg rotatesplit"));
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/clients");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 0,0");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 1920,540");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg rotatesplit"));
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/clients");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 960,0");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 960,1080");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg rotatesplit"));
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/clients");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 0,540");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 1920,540");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg rotatesplit"));
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/clients");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 0,0");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 960,1080");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// test different angles
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg rotatesplit 180"));
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/clients");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 960,0");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 960,1080");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg rotatesplit 270"));
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/clients");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 0,540");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 1920,540");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg rotatesplit 360"));
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/clients");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 0,0");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 1920,540");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// test negative angles
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg rotatesplit -90"));
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/clients");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 0,0");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 960,1080");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/dispatch layoutmsg rotatesplit -180"));
|
|
|
|
|
{
|
|
|
|
|
auto str = getFromSocket("/clients");
|
|
|
|
|
EXPECT_CONTAINS(str, "at: 960,0");
|
|
|
|
|
EXPECT_CONTAINS(str, "size: 960,1080");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
NLog::log("{}Killing all windows", Colors::YELLOW);
|
|
|
|
|
Tests::killAllWindows();
|
|
|
|
|
|
|
|
|
|
OK(getFromSocket("/reload"));
|
|
|
|
|
}
|
|
|
|
|
|
2025-10-29 23:21:28 +00:00
|
|
|
static bool test() {
|
|
|
|
|
NLog::log("{}Testing Dwindle layout", Colors::GREEN);
|
|
|
|
|
|
|
|
|
|
// test
|
|
|
|
|
NLog::log("{}Testing float clamp", Colors::GREEN);
|
|
|
|
|
testFloatClamp();
|
|
|
|
|
|
2026-02-26 12:00:05 +00:00
|
|
|
NLog::log("{}Testing #13349", Colors::GREEN);
|
|
|
|
|
test13349();
|
|
|
|
|
|
2026-03-02 16:21:20 +00:00
|
|
|
NLog::log("{}Testing splits", Colors::GREEN);
|
|
|
|
|
testSplit();
|
|
|
|
|
|
2026-03-06 23:47:48 +03:00
|
|
|
NLog::log("{}Testing rotatesplit", Colors::GREEN);
|
|
|
|
|
testRotatesplit();
|
|
|
|
|
|
2025-10-29 23:21:28 +00:00
|
|
|
// clean up
|
|
|
|
|
NLog::log("Cleaning up", Colors::YELLOW);
|
|
|
|
|
getFromSocket("/dispatch workspace 1");
|
|
|
|
|
OK(getFromSocket("/reload"));
|
|
|
|
|
|
|
|
|
|
return !ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
REGISTER_TEST_FN(test);
|