oh, 2 things in this commit..

1) THIS ONE IS IMPORTANT: don't set client->iconfied if the window is not actually going to be allowed to iconify. heh!

2) changes to focus fallback to avoid crashing and to avoid losing focus all at the same time.
This commit is contained in:
Dana Jansens 2007-03-14 15:51:21 +00:00
parent 768ce7b456
commit adc5675823
8 changed files with 67 additions and 78 deletions

View file

@ -1172,7 +1172,7 @@ void action_focus(union ActionData *data)
void action_unfocus (union ActionData *data) void action_unfocus (union ActionData *data)
{ {
if (data->client.any.c == focus_client) if (data->client.any.c == focus_client)
focus_fallback(OB_FOCUS_FALLBACK_UNFOCUSING); focus_fallback(FALSE);
} }
void action_iconify(union ActionData *data) void action_iconify(union ActionData *data)

View file

@ -78,6 +78,7 @@ static void client_apply_startup_state(ObClient *self, gint x, gint y);
static void client_restore_session_state(ObClient *self); static void client_restore_session_state(ObClient *self);
static void client_restore_session_stacking(ObClient *self); static void client_restore_session_stacking(ObClient *self);
static ObAppSettings *client_get_settings_state(ObClient *self); static ObAppSettings *client_get_settings_state(ObClient *self);
static void client_unfocus(ObClient *self);
void client_startup(gboolean reconfig) void client_startup(gboolean reconfig)
{ {
@ -2515,10 +2516,10 @@ static void client_iconify_recursive(ObClient *self,
ob_debug("%sconifying window: 0x%lx\n", (iconic ? "I" : "Uni"), ob_debug("%sconifying window: 0x%lx\n", (iconic ? "I" : "Uni"),
self->window); self->window);
self->iconic = iconic;
if (iconic) { if (iconic) {
if (self->functions & OB_CLIENT_FUNC_ICONIFY) { if (self->functions & OB_CLIENT_FUNC_ICONIFY) {
self->iconic = iconic;
/* update the focus lists.. iconic windows go to the bottom of /* update the focus lists.. iconic windows go to the bottom of
the list, put the new iconic window at the 'top of the the list, put the new iconic window at the 'top of the
bottom'. */ bottom'. */
@ -2527,6 +2528,8 @@ static void client_iconify_recursive(ObClient *self,
changed = TRUE; changed = TRUE;
} }
} else { } else {
self->iconic = iconic;
if (curdesk) if (curdesk)
client_set_desktop(self, screen_desktop, FALSE); client_set_desktop(self, screen_desktop, FALSE);
@ -3031,14 +3034,7 @@ gboolean client_focus(ObClient *self)
ob_debug("Focusing client \"%s\" at time %u\n", self->title, event_curtime); ob_debug("Focusing client \"%s\" at time %u\n", self->title, event_curtime);
if (self->can_focus) { if (self->can_focus) {
/* RevertToPointerRoot causes much more headache than RevertToNone, so XSetInputFocus(ob_display, self->window, RevertToPointerRoot,
I choose to use it always, hopefully to find errors quicker, if any
are left. (I hate X. I hate focus events.)
Update: Changing this to RevertToNone fixed a bug with mozilla (bug
#799. So now it is RevertToNone again.
*/
XSetInputFocus(ob_display, self->window, RevertToNone,
event_curtime); event_curtime);
} }
@ -3073,13 +3069,13 @@ gboolean client_focus(ObClient *self)
/* Used when the current client is closed or otherwise hidden, focus_last will /* Used when the current client is closed or otherwise hidden, focus_last will
then prevent focus from going to the mouse pointer then prevent focus from going to the mouse pointer
*/ */
void client_unfocus(ObClient *self) static void client_unfocus(ObClient *self)
{ {
if (focus_client == self) { if (focus_client == self) {
#ifdef DEBUG_FOCUS #ifdef DEBUG_FOCUS
ob_debug("client_unfocus for %lx\n", self->window); ob_debug("client_unfocus for %lx\n", self->window);
#endif #endif
focus_fallback(OB_FOCUS_FALLBACK_CLOSED); focus_fallback(FALSE);
} }
} }

View file

@ -494,9 +494,6 @@ gboolean client_can_focus(ObClient *self);
*/ */
gboolean client_focus(ObClient *self); gboolean client_focus(ObClient *self);
/*! Remove focus from the client window */
void client_unfocus(ObClient *self);
/*! Activates the client for use, focusing, uniconifying it, etc. To be used /*! Activates the client for use, focusing, uniconifying it, etc. To be used
when the user deliberately selects a window for use. when the user deliberately selects a window for use.
@param here If true, then the client is brought to the current desktop; @param here If true, then the client is brought to the current desktop;

View file

@ -422,7 +422,7 @@ static void event_process(const XEvent *ec, gpointer data)
} }
} }
#if 0 /* focus debugging stuff */ #if 1 /* focus debugging stuff */
if (e->type == FocusIn || e->type == FocusOut) { if (e->type == FocusIn || e->type == FocusOut) {
gint mode = e->xfocus.mode; gint mode = e->xfocus.mode;
gint detail = e->xfocus.detail; gint detail = e->xfocus.detail;
@ -655,8 +655,12 @@ static void event_handle_client(ObClient *client, XEvent *e)
case FocusOut: case FocusOut:
/* Look for the followup FocusIn */ /* Look for the followup FocusIn */
if (!XCheckIfEvent(ob_display, &ce, look_for_focusin, NULL)) { if (!XCheckIfEvent(ob_display, &ce, look_for_focusin, NULL)) {
/* There is no FocusIn, move focus where we can still hear events*/ /* There is no FocusIn, this means focus went to a window that
focus_fallback(OB_FOCUS_FALLBACK_NOFOCUS); is not being managed. most likely, this went to PointerRoot
or None, meaning the window is no longer around so fallback
focus, but not to that window */
ob_debug("Focus went to a black hole !\n");
focus_fallback(FALSE);
} else if (ce.xany.window == e->xany.window) { } else if (ce.xany.window == e->xany.window) {
/* If focus didn't actually move anywhere, there is nothing to do*/ /* If focus didn't actually move anywhere, there is nothing to do*/
break; break;
@ -667,7 +671,9 @@ static void event_handle_client(ObClient *client, XEvent *e)
if (ed.ignored) { if (ed.ignored) {
/* The FocusIn was ignored, this means it was on a window /* The FocusIn was ignored, this means it was on a window
that isn't a client. */ that isn't a client. */
focus_fallback(OB_FOCUS_FALLBACK_NOFOCUS); ob_debug("Focus went to an unmanaged window 0x%x !\n",
ce.xfocus.window);
focus_fallback(TRUE);
} }
} }

View file

@ -248,7 +248,7 @@ static ObClient* focus_fallback_transient(ObClient *top, ObClient *old)
return NULL; return NULL;
} }
ObClient* focus_fallback_target(ObFocusFallbackType type) ObClient* focus_fallback_target(gboolean allow_refocus)
{ {
GList *it; GList *it;
ObClient *old; ObClient *old;
@ -256,60 +256,57 @@ ObClient* focus_fallback_target(ObFocusFallbackType type)
old = focus_client; old = focus_client;
if ((type == OB_FOCUS_FALLBACK_UNFOCUSING if (!allow_refocus && old && old->transient_for) {
|| type == OB_FOCUS_FALLBACK_CLOSED) && old) { gboolean trans = FALSE;
if (old->transient_for) {
gboolean trans = FALSE;
if (!config_focus_follow || config_focus_last) if (!config_focus_follow || config_focus_last)
trans = TRUE; trans = TRUE;
else if ((target = client_under_pointer()) && else if ((target = client_under_pointer()) &&
client_search_transient client_search_transient
(client_search_top_parent(target), old)) (client_search_top_parent(target), old))
trans = TRUE; trans = TRUE;
/* try for transient relations */ /* try for transient relations */
if (trans) { if (trans) {
if (old->transient_for == OB_TRAN_GROUP) { if (old->transient_for == OB_TRAN_GROUP) {
for (it = focus_order[screen_desktop]; it; for (it = focus_order[screen_desktop]; it;
it = g_list_next(it)) it = g_list_next(it))
{
GSList *sit;
for (sit = old->group->members; sit;
sit = g_slist_next(sit))
{ {
GSList *sit; if (sit->data == it->data)
if ((target =
for (sit = old->group->members; sit; focus_fallback_transient(sit->data, old)))
sit = g_slist_next(sit)) {
{ ob_debug("found in transient #1\n");
if (sit->data == it->data) return target;
if ((target = }
focus_fallback_transient(sit->data, old)))
{
ob_debug("found in transient #1\n");
return target;
}
}
}
} else {
if ((target =
focus_fallback_transient(old->transient_for, old)))
{
ob_debug("found in transient #2\n");
return target;
} }
} }
} else {
if ((target =
focus_fallback_transient(old->transient_for, old)))
{
ob_debug("found in transient #2\n");
return target;
}
} }
} }
} }
ob_debug("trying pointer stuff\n"); ob_debug("trying pointer stuff\n");
if (config_focus_follow && if (config_focus_follow && !config_focus_last)
(type == OB_FOCUS_FALLBACK_UNFOCUSING || !config_focus_last))
{ {
if ((target = client_under_pointer())) if ((target = client_under_pointer()))
if (client_normal(target) && client_can_focus(target) && if (allow_refocus || target != old)
client_validate(target)) { if (client_normal(target) && client_can_focus(target) &&
ob_debug("found in pointer stuff\n"); client_validate(target)) {
return target; ob_debug("found in pointer stuff\n");
} return target;
}
} }
#if 0 #if 0
@ -328,7 +325,7 @@ ObClient* focus_fallback_target(ObFocusFallbackType type)
ob_debug("trying the focus order\n"); ob_debug("trying the focus order\n");
for (it = focus_order[screen_desktop]; it; it = g_list_next(it)) for (it = focus_order[screen_desktop]; it; it = g_list_next(it))
if (type != OB_FOCUS_FALLBACK_UNFOCUSING || it->data != old) if (allow_refocus || it->data != old)
if (client_normal(it->data) && client_can_focus(it->data) && if (client_normal(it->data) && client_can_focus(it->data) &&
client_validate(it->data)) client_validate(it->data))
{ {
@ -344,7 +341,7 @@ ObClient* focus_fallback_target(ObFocusFallbackType type)
return NULL; return NULL;
} }
void focus_fallback(ObFocusFallbackType type) void focus_fallback(gboolean allow_refocus)
{ {
ObClient *new; ObClient *new;
@ -354,7 +351,7 @@ void focus_fallback(ObFocusFallbackType type)
*/ */
focus_set_client(NULL); focus_set_client(NULL);
if ((new = focus_fallback_target(type))) if ((new = focus_fallback_target(allow_refocus)))
client_focus(new); client_focus(new);
} }

View file

@ -50,17 +50,10 @@ void focus_shutdown(gboolean reconfig);
send focus anywhere, its called by the Focus event handlers */ send focus anywhere, its called by the Focus event handlers */
void focus_set_client(struct _ObClient *client); void focus_set_client(struct _ObClient *client);
typedef enum { struct _ObClient* focus_fallback_target(gboolean allow_refocus);
OB_FOCUS_FALLBACK_UNFOCUSING, /*!< forcefully remove focus from the
current window */
OB_FOCUS_FALLBACK_CLOSED, /*!< closed the window with focus */
OB_FOCUS_FALLBACK_NOFOCUS /*!< nothing has focus for some reason */
} ObFocusFallbackType;
struct _ObClient* focus_fallback_target(ObFocusFallbackType type);
/*! Call this when you need to focus something! */ /*! Call this when you need to focus something! */
void focus_fallback(ObFocusFallbackType type); void focus_fallback(gboolean allow_refocus);
/*! Cycle focus amongst windows. */ /*! Cycle focus amongst windows. */
void focus_cycle(gboolean forward, gboolean linear, gboolean interactive, void focus_cycle(gboolean forward, gboolean linear, gboolean interactive,

View file

@ -278,7 +278,7 @@ gint main(gint argc, gchar **argv)
if (!reconfigure) { if (!reconfigure) {
/* get all the existing windows */ /* get all the existing windows */
client_manage_all(); client_manage_all();
focus_fallback(OB_FOCUS_FALLBACK_NOFOCUS); focus_fallback(TRUE);
} else { } else {
GList *it; GList *it;

View file

@ -463,7 +463,7 @@ void screen_set_desktop(guint num)
event_ignore_queued_enters(); event_ignore_queued_enters();
focus_hilite = focus_fallback_target(OB_FOCUS_FALLBACK_NOFOCUS); focus_hilite = focus_fallback_target(TRUE);
if (focus_hilite) { if (focus_hilite) {
frame_adjust_focus(focus_hilite->frame, TRUE); frame_adjust_focus(focus_hilite->frame, TRUE);
@ -893,7 +893,7 @@ void screen_show_desktop(gboolean show)
client_validate(it->data) && client_focus(it->data)) client_validate(it->data) && client_focus(it->data))
break; break;
} else { } else {
focus_fallback(OB_FOCUS_FALLBACK_NOFOCUS); focus_fallback(TRUE);
} }
show = !!show; /* make it boolean */ show = !!show; /* make it boolean */