Merge pull request #1664 from Airblader/feature-wm-state-hidden-2
Set _NET_WM_STATE_HIDDEN in tabbed/stacked containers
This commit is contained in:
commit
79d54fc3ae
|
@ -5,6 +5,7 @@ xmacro(_NET_WM_MOVERESIZE)
|
||||||
xmacro(_NET_WM_STATE_FULLSCREEN)
|
xmacro(_NET_WM_STATE_FULLSCREEN)
|
||||||
xmacro(_NET_WM_STATE_DEMANDS_ATTENTION)
|
xmacro(_NET_WM_STATE_DEMANDS_ATTENTION)
|
||||||
xmacro(_NET_WM_STATE_MODAL)
|
xmacro(_NET_WM_STATE_MODAL)
|
||||||
|
xmacro(_NET_WM_STATE_HIDDEN)
|
||||||
xmacro(_NET_WM_STATE)
|
xmacro(_NET_WM_STATE)
|
||||||
xmacro(_NET_WM_WINDOW_TYPE)
|
xmacro(_NET_WM_WINDOW_TYPE)
|
||||||
xmacro(_NET_WM_WINDOW_TYPE_NORMAL)
|
xmacro(_NET_WM_WINDOW_TYPE_NORMAL)
|
||||||
|
|
|
@ -42,12 +42,19 @@ bool con_is_leaf(Con *con);
|
||||||
*/
|
*/
|
||||||
bool con_has_managed_window(Con *con);
|
bool con_has_managed_window(Con *con);
|
||||||
|
|
||||||
/*
|
/**
|
||||||
* Returns true if a container should be considered split.
|
* Returns true if a container should be considered split.
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
bool con_is_split(Con *con);
|
bool con_is_split(Con *con);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This will only return true for containers which have some parent with
|
||||||
|
* a tabbed / stacked parent of which they are not the currently focused child.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
bool con_is_hidden(Con *con);
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Returns true if this node has regular or floating children.
|
* Returns true if this node has regular or floating children.
|
||||||
*
|
*
|
||||||
|
|
23
src/con.c
23
src/con.c
|
@ -260,6 +260,29 @@ bool con_is_split(Con *con) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* This will only return true for containers which have some parent with
|
||||||
|
* a tabbed / stacked parent of which they are not the currently focused child.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
bool con_is_hidden(Con *con) {
|
||||||
|
Con *current = con;
|
||||||
|
|
||||||
|
/* ascend to the workspace level and memorize the highest-up container
|
||||||
|
* which is stacked or tabbed. */
|
||||||
|
while (current != NULL && current->type != CT_WORKSPACE) {
|
||||||
|
Con *parent = current->parent;
|
||||||
|
if (parent != NULL && (parent->layout == L_TABBED || parent->layout == L_STACKED)) {
|
||||||
|
if (TAILQ_FIRST(&(parent->focus_head)) != current)
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
current = parent;
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Returns true if this node accepts a window (if the node swallows windows,
|
* Returns true if this node accepts a window (if the node swallows windows,
|
||||||
* it might already have swallowed enough and cannot hold any more).
|
* it might already have swallowed enough and cannot hold any more).
|
||||||
|
|
|
@ -234,6 +234,6 @@ void ewmh_setup_hints(void) {
|
||||||
/* I’m not entirely sure if we need to keep _NET_WM_NAME on root. */
|
/* I’m not entirely sure if we need to keep _NET_WM_NAME on root. */
|
||||||
xcb_change_property(conn, XCB_PROP_MODE_REPLACE, root, A__NET_WM_NAME, A_UTF8_STRING, 8, strlen("i3"), "i3");
|
xcb_change_property(conn, XCB_PROP_MODE_REPLACE, root, A__NET_WM_NAME, A_UTF8_STRING, 8, strlen("i3"), "i3");
|
||||||
|
|
||||||
/* only send the first 29 atoms (last one is _NET_CLOSE_WINDOW) increment that number when adding supported atoms */
|
/* only send the first 30 atoms (last one is _NET_CLOSE_WINDOW) increment that number when adding supported atoms */
|
||||||
xcb_change_property(conn, XCB_PROP_MODE_REPLACE, root, A__NET_SUPPORTED, XCB_ATOM_ATOM, 32, 29, supported_atoms);
|
xcb_change_property(conn, XCB_PROP_MODE_REPLACE, root, A__NET_SUPPORTED, XCB_ATOM_ATOM, 32, 30, supported_atoms);
|
||||||
}
|
}
|
||||||
|
|
30
src/x.c
30
src/x.c
|
@ -36,6 +36,7 @@ typedef struct con_state {
|
||||||
bool mapped;
|
bool mapped;
|
||||||
bool unmap_now;
|
bool unmap_now;
|
||||||
bool child_mapped;
|
bool child_mapped;
|
||||||
|
bool is_hidden;
|
||||||
|
|
||||||
/** The con for which this state is. */
|
/** The con for which this state is. */
|
||||||
Con *con;
|
Con *con;
|
||||||
|
@ -611,6 +612,33 @@ void x_deco_recurse(Con *con) {
|
||||||
x_draw_decoration(con);
|
x_draw_decoration(con);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Sets or removes the _NET_WM_STATE_HIDDEN property on con if necessary.
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
static void set_hidden_state(Con *con) {
|
||||||
|
if (con->window == NULL) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
con_state *state = state_for_frame(con->frame);
|
||||||
|
bool should_be_hidden = con_is_hidden(con);
|
||||||
|
if (should_be_hidden == state->is_hidden)
|
||||||
|
return;
|
||||||
|
|
||||||
|
unsigned int num = 0;
|
||||||
|
uint32_t values[1];
|
||||||
|
if (should_be_hidden) {
|
||||||
|
DLOG("setting _NET_WM_STATE_HIDDEN for con = %p\n", con);
|
||||||
|
values[num++] = A__NET_WM_STATE_HIDDEN;
|
||||||
|
} else {
|
||||||
|
DLOG("removing _NET_WM_STATE_HIDDEN for con = %p\n", con);
|
||||||
|
}
|
||||||
|
|
||||||
|
xcb_change_property(conn, XCB_PROP_MODE_REPLACE, con->window->id, A__NET_WM_STATE, XCB_ATOM_ATOM, 32, num, values);
|
||||||
|
state->is_hidden = should_be_hidden;
|
||||||
|
}
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* This function pushes the properties of each node of the layout tree to
|
* This function pushes the properties of each node of the layout tree to
|
||||||
* X11 if they have changed (like the map state, position of the window, …).
|
* X11 if they have changed (like the map state, position of the window, …).
|
||||||
|
@ -814,6 +842,8 @@ void x_push_node(Con *con) {
|
||||||
fake_absolute_configure_notify(con);
|
fake_absolute_configure_notify(con);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
set_hidden_state(con);
|
||||||
|
|
||||||
/* Handle all children and floating windows of this node. We recurse
|
/* Handle all children and floating windows of this node. We recurse
|
||||||
* in focus order to display the focused client in a stack first when
|
* in focus order to display the focused client in a stack first when
|
||||||
* switching workspaces (reduces flickering). */
|
* switching workspaces (reduces flickering). */
|
||||||
|
|
|
@ -0,0 +1,201 @@
|
||||||
|
#!perl
|
||||||
|
# vim:ts=4:sw=4:expandtab
|
||||||
|
#
|
||||||
|
# Please read the following documents before working on tests:
|
||||||
|
# • http://build.i3wm.org/docs/testsuite.html
|
||||||
|
# (or docs/testsuite)
|
||||||
|
#
|
||||||
|
# • http://build.i3wm.org/docs/lib-i3test.html
|
||||||
|
# (alternatively: perldoc ./testcases/lib/i3test.pm)
|
||||||
|
#
|
||||||
|
# • http://build.i3wm.org/docs/ipc.html
|
||||||
|
# (or docs/ipc)
|
||||||
|
#
|
||||||
|
# • http://onyxneon.com/books/modern_perl/modern_perl_a4.pdf
|
||||||
|
# (unless you are already familiar with Perl)
|
||||||
|
#
|
||||||
|
# Tests for setting and removing the _NET_WM_STATE_HIDDEN atom properly.
|
||||||
|
# Ticket: #1648
|
||||||
|
use i3test;
|
||||||
|
use X11::XCB qw(:all);
|
||||||
|
|
||||||
|
sub is_hidden {
|
||||||
|
sync_with_i3;
|
||||||
|
my $atom = $x->atom(name => '_NET_WM_STATE_HIDDEN');
|
||||||
|
|
||||||
|
my ($con) = @_;
|
||||||
|
my $cookie = $x->get_property(
|
||||||
|
0,
|
||||||
|
$con->{id},
|
||||||
|
$x->atom(name => '_NET_WM_STATE')->id,
|
||||||
|
GET_PROPERTY_TYPE_ANY,
|
||||||
|
0,
|
||||||
|
4096
|
||||||
|
);
|
||||||
|
|
||||||
|
my $reply = $x->get_property_reply($cookie->{sequence});
|
||||||
|
my $len = $reply->{length};
|
||||||
|
return 0 if $len == 0;
|
||||||
|
|
||||||
|
my @atoms = unpack("L$len", $reply->{value});
|
||||||
|
for (my $i = 0; $i < $len; $i++) {
|
||||||
|
return 1 if $atoms[$i] == $atom->id;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
my ($tabA, $tabB, $tabC, $subtabA, $subtabB, $windowA, $windowB);
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Given two containers next to each other, when focusing one, then the other
|
||||||
|
# one does not have _NET_WM_STATE_HIDDEN set.
|
||||||
|
###############################################################################
|
||||||
|
|
||||||
|
fresh_workspace;
|
||||||
|
$windowA = open_window;
|
||||||
|
$windowB = open_window;
|
||||||
|
|
||||||
|
ok(!is_hidden($windowA), 'left window does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(!is_hidden($windowB), 'right window does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Given two containers on different workspaces, when one is focused, then
|
||||||
|
# the other one does not have _NET_WM_STATE_HIDDEN set.
|
||||||
|
###############################################################################
|
||||||
|
|
||||||
|
fresh_workspace;
|
||||||
|
$windowA = open_window;
|
||||||
|
fresh_workspace;
|
||||||
|
$windowB = open_window;
|
||||||
|
|
||||||
|
ok(!is_hidden($windowA), 'left window does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(!is_hidden($windowB), 'right window does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Given two containers in the same tabbed container, when one is focused, then
|
||||||
|
# (only) the other one has _NET_WM_STATE_HIDDEN set.
|
||||||
|
# Given the other tab is focused, then the atom is transferred.
|
||||||
|
###############################################################################
|
||||||
|
|
||||||
|
fresh_workspace;
|
||||||
|
$tabA = open_window;
|
||||||
|
cmd 'layout tabbed';
|
||||||
|
$tabB = open_window;
|
||||||
|
|
||||||
|
ok(is_hidden($tabA), 'unfocused tab has _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(!is_hidden($tabB), 'focused tab does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
|
||||||
|
cmd 'focus left';
|
||||||
|
|
||||||
|
ok(!is_hidden($tabA), 'focused tab does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(is_hidden($tabB), 'unfocused tab has _NET_WM_STATE_HIDDEN set');
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Given three containers in the same stacked container, when the focused tab
|
||||||
|
# is moved to another workspace, then the now focused tab does not have
|
||||||
|
# _NET_WM_STATE_HIDDEN set anymore.
|
||||||
|
###############################################################################
|
||||||
|
|
||||||
|
fresh_workspace;
|
||||||
|
$tabA = open_window;
|
||||||
|
cmd 'layout stacked';
|
||||||
|
$tabB = open_window;
|
||||||
|
$tabC = open_window;
|
||||||
|
cmd 'move window to workspace unused';
|
||||||
|
|
||||||
|
ok(is_hidden($tabA), 'unfocused tab has _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(!is_hidden($tabB), 'focused tab does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(!is_hidden($tabC), 'moved window does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Given three containers in the same stacked container, when a not focused
|
||||||
|
# tab is moved to another workspace, then it does not have _NET_WM_STATE_HIDDEN
|
||||||
|
# set anymore.
|
||||||
|
###############################################################################
|
||||||
|
|
||||||
|
fresh_workspace;
|
||||||
|
$tabA = open_window;
|
||||||
|
cmd 'layout stacked';
|
||||||
|
$tabB = open_window;
|
||||||
|
cmd 'mark moveme';
|
||||||
|
$tabC = open_window;
|
||||||
|
cmd '[con_mark="moveme"] move window to workspace unused';
|
||||||
|
|
||||||
|
ok(is_hidden($tabA), 'unfocused tab has _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(!is_hidden($tabB), 'moved window does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(!is_hidden($tabC), 'focused tab does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Given a tabbed container and some other container, when the latter is moved
|
||||||
|
# into the tabbed container, then all other tabs have _NET_WM_STATE_HIDDEN
|
||||||
|
# set.
|
||||||
|
###############################################################################
|
||||||
|
|
||||||
|
fresh_workspace;
|
||||||
|
$tabA = open_window;
|
||||||
|
cmd 'layout tabbed';
|
||||||
|
$tabB = open_window;
|
||||||
|
cmd 'focus parent';
|
||||||
|
cmd 'split h';
|
||||||
|
$tabC = open_window;
|
||||||
|
cmd 'move left';
|
||||||
|
|
||||||
|
ok(is_hidden($tabA), 'unfocused tab has _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(is_hidden($tabB), 'unfocused tab has _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(!is_hidden($tabC), 'focused tab does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Given a stacked container nested inside another tabbed container with the
|
||||||
|
# inner one being in the currently focused tab, then the focused tab of the
|
||||||
|
# inner container does not have _NET_WM_STATE_HIDDEN set.
|
||||||
|
###############################################################################
|
||||||
|
|
||||||
|
fresh_workspace;
|
||||||
|
$tabA = open_window;
|
||||||
|
cmd 'layout tabbed';
|
||||||
|
$tabB = open_window;
|
||||||
|
cmd 'split h';
|
||||||
|
open_window;
|
||||||
|
cmd 'split v';
|
||||||
|
cmd 'layout stacked';
|
||||||
|
$subtabA = open_window;
|
||||||
|
$subtabB = open_window;
|
||||||
|
|
||||||
|
ok(is_hidden($tabA), 'unfocused outer tab has _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(!is_hidden($tabB), 'focused outer tab does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(is_hidden($subtabA), 'unfocused inner tab has _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(!is_hidden($subtabB), 'focused inner tab does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
|
||||||
|
cmd 'focus left';
|
||||||
|
|
||||||
|
ok(!is_hidden($subtabB), 'focused inner tab does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
# Given a stacked container nested inside another tabbed container with the
|
||||||
|
# inner one being in a currently not focused tab, then all tabs of the inner
|
||||||
|
# container have _NET_WM_STATE_HIDDEN set.
|
||||||
|
###############################################################################
|
||||||
|
|
||||||
|
fresh_workspace;
|
||||||
|
$tabA = open_window;
|
||||||
|
cmd 'layout tabbed';
|
||||||
|
$tabB = open_window;
|
||||||
|
cmd 'split h';
|
||||||
|
open_window;
|
||||||
|
cmd 'split v';
|
||||||
|
cmd 'layout stacked';
|
||||||
|
$subtabA = open_window;
|
||||||
|
$subtabB = open_window;
|
||||||
|
cmd 'focus parent';
|
||||||
|
cmd 'focus parent';
|
||||||
|
cmd 'focus left';
|
||||||
|
|
||||||
|
ok(!is_hidden($tabA), 'focused outer tab does not have _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(is_hidden($tabB), 'unfocused outer tab has _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(is_hidden($subtabA), 'unfocused inner tab has _NET_WM_STATE_HIDDEN set');
|
||||||
|
ok(is_hidden($subtabB), 'unfocused inner tab has _NET_WM_STATE_HIDDEN set');
|
||||||
|
|
||||||
|
###############################################################################
|
||||||
|
|
||||||
|
done_testing;
|
Loading…
Reference in New Issue