#!perl # vim:ts=4:sw=4:expandtab # # Tests if assignments work # use i3test i3_autostart => 0; use X11::XCB qw(PROP_MODE_REPLACE); # TODO: move to X11::XCB sub set_wm_class { my ($id, $class, $instance) = @_; # Add a _NET_WM_STRUT_PARTIAL hint my $atomname = $x->atom(name => 'WM_CLASS'); my $atomtype = $x->atom(name => 'STRING'); $x->change_property( PROP_MODE_REPLACE, $id, $atomname->id, $atomtype->id, 8, length($class) + length($instance) + 2, "$instance\x00$class\x00" ); } sub open_special { my %args = @_; my $wm_class = delete($args{wm_class}) || 'special'; $args{name} //= 'special window'; # We use dont_map because i3 will not map the window on the current # workspace. Thus, open_window would time out in wait_for_map (2 seconds). my $window = open_window( %args, before_map => sub { set_wm_class($_->id, $wm_class, $wm_class) }, dont_map => 1, ); $window->map; return $window; } ##################################################################### # start a window and see that it does not get assigned with an empty config ##################################################################### my $config = <destroy; ##################################################################### # start a window and see that it gets assigned to a formerly unused # workspace ##################################################################### $config = <destroy; exit_gracefully($pid); sleep 0.25; ##################################################################### # start a window and see that it gets assigned to a workspace which has content # already, next to the existing node. ##################################################################### $pid = launch_with_config($config); # initialize the target workspace, then go to a fresh one ok(!("targetws" ~~ @{get_workspace_names()}), 'targetws does not exist yet'); cmd 'workspace targetws'; cmp_ok(@{get_ws_content('targetws')}, '==', 0, 'no containers on targetws yet'); cmd 'open'; cmp_ok(@{get_ws_content('targetws')}, '==', 1, 'one container on targetws'); $tmp = fresh_workspace; ok(@{get_ws_content($tmp)} == 0, 'no containers yet'); ok("targetws" ~~ @{get_workspace_names()}, 'targetws does not exist yet'); # We use sync_with_i3 instead of wait_for_map here because i3 will not actually # map the window -- it will be assigned to a different workspace and will only # be mapped once you switch to that workspace $window = open_special(dont_map => 1); $window->map; sync_with_i3; ok(@{get_ws_content($tmp)} == 0, 'still no containers'); ok(@{get_ws_content('targetws')} == 2, 'two containers on targetws'); exit_gracefully($pid); ##################################################################### # start a window and see that it gets assigned to a workspace which has content # already, next to the existing node. ##################################################################### $config = <{nodes}} == 0, 'no tiling cons'); ok(@{$content->{floating_nodes}} == 1, 'one floating con'); $window->destroy; exit_gracefully($pid); sleep 0.25; ##################################################################### # make sure that assignments are case-insensitive in the old syntax. ##################################################################### $config = < 'SPEcial'); $content = get_ws($tmp); ok(@{$content->{nodes}} == 0, 'no tiling cons'); ok(@{$content->{floating_nodes}} == 1, 'one floating con'); $window->destroy; exit_gracefully($pid); sleep 0.25; ##################################################################### # regression test: dock clients with floating assignments should not crash # (instead, nothing should happen - dock clients can’t float) # ticket #501 ##################################################################### $config = < $x->atom(name => '_NET_WM_WINDOW_TYPE_DOCK'), ); $content = get_ws($tmp); ok(@{$content->{nodes}} == 0, 'no tiling cons'); ok(@{$content->{floating_nodes}} == 0, 'one floating con'); @docked = get_dock_clients; is(@docked, 2, 'two dock clients now'); $window->destroy; does_i3_live; exit_gracefully($pid); done_testing;