Argument for 'kill' for killing a specific window (now default) or the whole client (+test)

Use 'kill window' to kill a specific window (for example only one specific
popup), use 'kill client' to kill the whole application (or X11 connection to
be specific).
This commit is contained in:
Michael Stapelberg 2011-05-13 20:41:03 +02:00
parent 44c2555e67
commit 167bdd26b7
13 changed files with 111 additions and 25 deletions

View File

@ -43,6 +43,10 @@ typedef enum { D_LEFT, D_RIGHT, D_UP, D_DOWN } direction_t;
typedef enum { NO_ORIENTATION = 0, HORIZ, VERT } orientation_t; typedef enum { NO_ORIENTATION = 0, HORIZ, VERT } orientation_t;
typedef enum { BS_NORMAL = 0, BS_NONE = 1, BS_1PIXEL = 2 } border_style_t; typedef enum { BS_NORMAL = 0, BS_NONE = 1, BS_1PIXEL = 2 } border_style_t;
/** parameter to specify whether tree_close() and x_window_kill() should kill
* only this specific window or the whole X11 client */
typedef enum { DONT_KILL_WINDOW = 0, KILL_WINDOW = 1, KILL_CLIENT = 2 } kill_window_t;
enum { enum {
BIND_NONE = 0, BIND_NONE = 0,
BIND_SHIFT = XCB_MOD_MASK_SHIFT, /* (1 << 0) */ BIND_SHIFT = XCB_MOD_MASK_SHIFT, /* (1 << 0) */

View File

@ -56,7 +56,7 @@ void tree_render();
* Closes the current container using tree_close(). * Closes the current container using tree_close().
* *
*/ */
void tree_close_con(); void tree_close_con(kill_window_t kill_window);
/** /**
* Changes focus in the given way (next/previous) and given orientation * Changes focus in the given way (next/previous) and given orientation
@ -71,7 +71,7 @@ void tree_next(char way, orientation_t orientation);
* and the window is expected to kill itself. * and the window is expected to kill itself.
* *
*/ */
bool tree_close(Con *con, bool kill_window, bool dont_kill_parent); bool tree_close(Con *con, kill_window_t kill_window, bool dont_kill_parent);
/** /**
* Loads tree from ~/.i3/_restart.json (used for in-place restarts). * Loads tree from ~/.i3/_restart.json (used for in-place restarts).

View File

@ -52,7 +52,7 @@ bool window_supports_protocol(xcb_window_t window, xcb_atom_t atom);
* Kills the given X11 window using WM_DELETE_WINDOW (if supported). * Kills the given X11 window using WM_DELETE_WINDOW (if supported).
* *
*/ */
void x_window_kill(xcb_window_t window); void x_window_kill(xcb_window_t window, kill_window_t kill_window);
/** /**
* Draws the decoration of the given container onto its parent. * Draws the decoration of the given container onto its parent.

View File

@ -2,7 +2,7 @@
* vim:ts=4:sw=4:expandtab * vim:ts=4:sw=4:expandtab
* *
* i3 - an improved dynamic tiling window manager * i3 - an improved dynamic tiling window manager
* © 2009-2010 Michael Stapelberg and contributors (see also: LICENSE) * © 2009-2011 Michael Stapelberg and contributors (see also: LICENSE)
* *
* cmdparse.l: the lexer for commands you send to i3 (or bind on keys) * cmdparse.l: the lexer for commands you send to i3 (or bind on keys)
* *
@ -92,6 +92,8 @@ exit { return TOK_EXIT; }
reload { return TOK_RELOAD; } reload { return TOK_RELOAD; }
restart { return TOK_RESTART; } restart { return TOK_RESTART; }
kill { return TOK_KILL; } kill { return TOK_KILL; }
window { return TOK_WINDOW; }
client { return TOK_CLIENT; }
fullscreen { return TOK_FULLSCREEN; } fullscreen { return TOK_FULLSCREEN; }
global { return TOK_GLOBAL; } global { return TOK_GLOBAL; }
layout { return TOK_LAYOUT; } layout { return TOK_LAYOUT; }

View File

@ -91,7 +91,7 @@ char *parse_cmd(const char *new) {
%} %}
%expect 4 %expect 5
%error-verbose %error-verbose
%lex-param { struct context *context } %lex-param { struct context *context }
@ -107,6 +107,8 @@ char *parse_cmd(const char *new) {
%token TOK_RELOAD "reload" %token TOK_RELOAD "reload"
%token TOK_RESTART "restart" %token TOK_RESTART "restart"
%token TOK_KILL "kill" %token TOK_KILL "kill"
%token TOK_WINDOW "window"
%token TOK_CLIENT "client"
%token TOK_FULLSCREEN "fullscreen" %token TOK_FULLSCREEN "fullscreen"
%token TOK_GLOBAL "global" %token TOK_GLOBAL "global"
%token TOK_LAYOUT "layout" %token TOK_LAYOUT "layout"
@ -161,6 +163,7 @@ char *parse_cmd(const char *new) {
%type <number> resize_px %type <number> resize_px
%type <number> resize_way %type <number> resize_way
%type <number> resize_tiling %type <number> resize_tiling
%type <number> optional_kill_mode
%% %%
@ -398,24 +401,30 @@ focus:
; ;
kill: kill:
TOK_KILL TOK_KILL optional_kill_mode
{ {
owindow *current; owindow *current;
printf("killing!\n"); printf("killing!\n");
/* check if the match is empty, not if the result is empty */ /* check if the match is empty, not if the result is empty */
if (match_is_empty(&current_match)) if (match_is_empty(&current_match))
tree_close_con(); tree_close_con($2);
else { else {
TAILQ_FOREACH(current, &owindows, owindows) { TAILQ_FOREACH(current, &owindows, owindows) {
printf("matching: %p / %s\n", current->con, current->con->name); printf("matching: %p / %s\n", current->con, current->con->name);
tree_close(current->con, true, false); tree_close(current->con, $2, false);
} }
} }
} }
; ;
optional_kill_mode:
/* empty */ { $$ = KILL_WINDOW; }
| WHITESPACE TOK_WINDOW { $$ = KILL_WINDOW; }
| WHITESPACE TOK_CLIENT { $$ = KILL_CLIENT; }
;
workspace: workspace:
TOK_WORKSPACE WHITESPACE STR TOK_WORKSPACE WHITESPACE STR
{ {

View File

@ -885,7 +885,7 @@ static void con_on_remove_child(Con *con) {
int children = con_num_children(con); int children = con_num_children(con);
if (children == 0) { if (children == 0) {
DLOG("Container empty, closing\n"); DLOG("Container empty, closing\n");
tree_close(con, false, false); tree_close(con, DONT_KILL_WINDOW, false);
return; return;
} }
} }

View File

@ -85,7 +85,7 @@ void floating_enable(Con *con, bool automatic) {
/* check if the parent container is empty and close it if so */ /* check if the parent container is empty and close it if so */
if ((con->parent->type == CT_CON || con->parent->type == CT_FLOATING_CON) && con_num_children(con->parent) == 0) { if ((con->parent->type == CT_CON || con->parent->type == CT_FLOATING_CON) && con_num_children(con->parent) == 0) {
DLOG("Old container empty after setting this child to floating, closing\n"); DLOG("Old container empty after setting this child to floating, closing\n");
tree_close(con->parent, false, false); tree_close(con->parent, DONT_KILL_WINDOW, false);
} }
char *name; char *name;
@ -186,7 +186,7 @@ void floating_disable(Con *con, bool automatic) {
/* 2: kill parent container */ /* 2: kill parent container */
TAILQ_REMOVE(&(con->parent->parent->floating_head), con->parent, floating_windows); TAILQ_REMOVE(&(con->parent->parent->floating_head), con->parent, floating_windows);
TAILQ_REMOVE(&(con->parent->parent->focus_head), con->parent, focused); TAILQ_REMOVE(&(con->parent->parent->focus_head), con->parent, focused);
tree_close(con->parent, false, false); tree_close(con->parent, DONT_KILL_WINDOW, false);
/* 3: re-attach to the parent of the currently focused con on the workspace /* 3: re-attach to the parent of the currently focused con on the workspace
* this floating con was on */ * this floating con was on */

View File

@ -466,7 +466,7 @@ static int handle_unmap_notify_event(xcb_unmap_notify_event_t *event) {
return 1; return 1;
} }
tree_close(con, false, false); tree_close(con, DONT_KILL_WINDOW, false);
tree_render(); tree_render();
x_push_changes(croot); x_push_changes(croot);
return 1; return 1;

View File

@ -709,7 +709,7 @@ void randr_query_outputs() {
} }
DLOG("destroying disappearing con %p\n", output->con); DLOG("destroying disappearing con %p\n", output->con);
tree_close(output->con, false, true); tree_close(output->con, DONT_KILL_WINDOW, true);
DLOG("Done. Should be fine now\n"); DLOG("Done. Should be fine now\n");
output->con = NULL; output->con = NULL;
} }

View File

@ -99,7 +99,7 @@ static bool _is_con_mapped(Con *con) {
* and the window is expected to kill itself. * and the window is expected to kill itself.
* *
*/ */
bool tree_close(Con *con, bool kill_window, bool dont_kill_parent) { bool tree_close(Con *con, kill_window_t kill_window, bool dont_kill_parent) {
bool was_mapped = con->mapped; bool was_mapped = con->mapped;
Con *parent = con->parent; Con *parent = con->parent;
@ -133,8 +133,8 @@ bool tree_close(Con *con, bool kill_window, bool dont_kill_parent) {
} }
if (con->window != NULL) { if (con->window != NULL) {
if (kill_window) { if (kill_window != DONT_KILL_WINDOW) {
x_window_kill(con->window->id); x_window_kill(con->window->id, kill_window);
return false; return false;
} else { } else {
/* un-parent the window */ /* un-parent the window */
@ -165,7 +165,7 @@ bool tree_close(Con *con, bool kill_window, bool dont_kill_parent) {
if (con_is_floating(con)) { if (con_is_floating(con)) {
Con *ws = con_get_workspace(con); Con *ws = con_get_workspace(con);
DLOG("Container was floating, killing floating container\n"); DLOG("Container was floating, killing floating container\n");
tree_close(parent, false, false); tree_close(parent, DONT_KILL_WINDOW, false);
DLOG("parent container killed\n"); DLOG("parent container killed\n");
if (con == focused) { if (con == focused) {
DLOG("This is the focused container, i need to find another one to focus. I start looking at ws = %p\n", ws); DLOG("This is the focused container, i need to find another one to focus. I start looking at ws = %p\n", ws);
@ -192,7 +192,7 @@ bool tree_close(Con *con, bool kill_window, bool dont_kill_parent) {
} }
if (was_mapped || con == focused) { if (was_mapped || con == focused) {
if (kill_window || !dont_kill_parent || con == focused) { if ((kill_window != DONT_KILL_WINDOW) || !dont_kill_parent || con == focused) {
DLOG("focusing %p / %s\n", next, next->name); DLOG("focusing %p / %s\n", next, next->name);
/* TODO: check if the container (or one of its children) was focused */ /* TODO: check if the container (or one of its children) was focused */
if (next->type == CT_DOCKAREA) { if (next->type == CT_DOCKAREA) {
@ -220,7 +220,7 @@ bool tree_close(Con *con, bool kill_window, bool dont_kill_parent) {
* Closes the current container using tree_close(). * Closes the current container using tree_close().
* *
*/ */
void tree_close_con() { void tree_close_con(kill_window_t kill_window) {
assert(focused != NULL); assert(focused != NULL);
if (focused->type == CT_WORKSPACE) { if (focused->type == CT_WORKSPACE) {
LOG("Cannot close workspace\n"); LOG("Cannot close workspace\n");
@ -232,7 +232,7 @@ void tree_close_con() {
assert(focused->type != CT_ROOT); assert(focused->type != CT_ROOT);
/* Kill con */ /* Kill con */
tree_close(focused, true, false); tree_close(focused, kill_window, false);
} }
/* /*
@ -463,7 +463,7 @@ void tree_flatten(Con *con) {
/* 4: close the redundant cons */ /* 4: close the redundant cons */
DLOG("closing redundant cons\n"); DLOG("closing redundant cons\n");
tree_close(con, false, true); tree_close(con, DONT_KILL_WINDOW, true);
/* Well, we got to abort the recursion here because we destroyed the /* Well, we got to abort the recursion here because we destroyed the
* container. However, if tree_flatten() is called sufficiently often, * container. However, if tree_flatten() is called sufficiently often,

View File

@ -244,7 +244,7 @@ void workspace_show(const char *num) {
/* check if this workspace is currently visible */ /* check if this workspace is currently visible */
if (!workspace_is_visible(old)) { if (!workspace_is_visible(old)) {
LOG("Closing old workspace (%p / %s), it is empty\n", old, old->name); LOG("Closing old workspace (%p / %s), it is empty\n", old, old->name);
tree_close(old, false, false); tree_close(old, DONT_KILL_WINDOW, false);
ipc_send_event("workspace", I3_IPC_EVENT_WORKSPACE, "{\"change\":\"empty\"}"); ipc_send_event("workspace", I3_IPC_EVENT_WORKSPACE, "{\"change\":\"empty\"}");
changed_num_workspaces = true; changed_num_workspaces = true;
} }

View File

@ -199,11 +199,16 @@ bool window_supports_protocol(xcb_window_t window, xcb_atom_t atom) {
* Kills the given X11 window using WM_DELETE_WINDOW (if supported). * Kills the given X11 window using WM_DELETE_WINDOW (if supported).
* *
*/ */
void x_window_kill(xcb_window_t window) { void x_window_kill(xcb_window_t window, kill_window_t kill_window) {
/* if this window does not support WM_DELETE_WINDOW, we kill it the hard way */ /* if this window does not support WM_DELETE_WINDOW, we kill it the hard way */
if (!window_supports_protocol(window, A_WM_DELETE_WINDOW)) { if (!window_supports_protocol(window, A_WM_DELETE_WINDOW)) {
LOG("Killing window the hard way\n"); if (kill_window == KILL_WINDOW) {
LOG("Killing specific window 0x%08x\n", window);
xcb_destroy_window(conn, window);
} else {
LOG("Killing the X11 client which owns window 0x%08x\n", window);
xcb_kill_client(conn, window); xcb_kill_client(conn, window);
}
return; return;
} }

View File

@ -0,0 +1,66 @@
#!perl
# vim:ts=4:sw=4:expandtab
#
# Tests if WM_STATE is WM_STATE_NORMAL when mapped and WM_STATE_WITHDRAWN when
# unmapped.
#
use X11::XCB qw(:all);
use X11::XCB::Connection;
use i3test;
my $x = X11::XCB::Connection->new;
sub two_windows {
my $tmp = fresh_workspace;
ok(@{get_ws_content($tmp)} == 0, 'no containers yet');
my $first = open_standard_window($x);
my $second = open_standard_window($x);
is($x->input_focus, $second->id, 'second window focused');
ok(@{get_ws_content($tmp)} == 2, 'two containers opened');
return $tmp;
}
##############################################################
# 1: open two windows (in the same client), kill one and see if
# the other one is still there
##############################################################
my $tmp = two_windows;
cmd 'kill';
sleep 0.25;
ok(@{get_ws_content($tmp)} == 1, 'one container left after killing');
##############################################################
# 2: same test case as test 1, but with the explicit variant
# 'kill window'
##############################################################
my $tmp = two_windows;
cmd 'kill window';
sleep 0.25;
ok(@{get_ws_content($tmp)} == 1, 'one container left after killing');
##############################################################
# 3: open two windows (in the same client), use 'kill client'
# and check if both are gone
##############################################################
my $tmp = two_windows;
cmd 'kill client';
sleep 0.25;
ok(@{get_ws_content($tmp)} == 0, 'no containers left after killing');
done_testing;