1 #include "../kernel/dispatch.h"
2 #include "../kernel/screen.h"
3 #include "../kernel/client.h"
4 #include "../kernel/frame.h"
5 #include "../kernel/focus.h"
6 #include "../kernel/stacking.h"
7 #include "../kernel/openbox.h"
10 static gboolean follow_mouse
= TRUE
;
11 static gboolean warp_on_desk_switch
= FALSE
;
12 static gboolean focus_new
= TRUE
;
14 /*static int skip_enter = 0;*/
16 static gboolean
focus_under_pointer()
23 if (XQueryPointer(ob_display
, ob_root
, &w
, &w
, &x
, &y
, &i
, &i
, &u
))
25 for (it
= stacking_list
; it
!= NULL
; it
= it
->next
) {
27 if (c
->desktop
== screen_desktop
&&
28 RECT_CONTAINS(c
->frame
->area
, x
, y
))
32 return client_normal(it
->data
) && client_focus(it
->data
);
38 static void chew_enter_events()
43 skip the next enter event from the desktop switch so focus
44 doesn't skip briefly to what was under the pointer */
46 /* kill all enter events from prior to the desktop switch, we
47 aren't interested in them if we have found our own target
49 XXX this is rude to other plugins...can this be done
50 better? count the events in the queue? */
51 while (XCheckTypedEvent(ob_display
, EnterNotify
, &e
));
54 XPutBackEvent(ob_display, &e);
61 static void focus_fallback(gboolean switching_desks
)
65 for (it
= focus_order
[screen_desktop
]; it
!= NULL
; it
= it
->next
)
66 if (client_normal(it
->data
) && client_focus(it
->data
)) {
67 if (switching_desks
) {
72 if (warp_on_desk_switch
) {
73 /* I have to do this warp twice! Otherwise windows dont get
74 Enter/Leave events when i warp on a desktop switch! */
75 XWarpPointer(ob_display
, None
, c
->window
, 0, 0, 0, 0,
76 c
->area
.width
/ 2, c
->area
.height
/ 2);
77 XWarpPointer(ob_display
, None
, c
->window
, 0, 0, 0, 0,
78 c
->area
.width
/ 2, c
->area
.height
/ 2);
85 static void focus_desktop()
89 for (it
= g_list_last(stacking_list
); it
!= NULL
; it
= it
->prev
) {
90 Client
*client
= it
->data
;
91 if (client
->type
== Type_Desktop
&& client
->frame
->visible
)
92 if (client_focus(client
))
98 static void event(ObEvent
*e
, void *foo
)
101 case Event_Client_Mapped
:
102 if (focus_new
&& client_normal(e
->data
.c
.client
))
103 client_focus(e
->data
.c
.client
);
106 case Event_Client_Unmapped
:
107 if (ob_state
== State_Exiting
) break;
109 if (client_focused(e
->data
.c
.client
))
110 if (!follow_mouse
|| !focus_under_pointer())
111 focus_fallback(FALSE
);
114 case Event_Client_Desktop
:
115 /* focus the next available target if moving from the current
117 if ((unsigned)e
->data
.c
.num
[1] == screen_desktop
)
118 if (!follow_mouse
|| !focus_under_pointer())
119 focus_fallback(FALSE
);
121 case Event_Ob_Desktop
:
122 focus_fallback(TRUE
);
125 case Event_Ob_ShowDesktop
:
126 if (!e
->data
.o
.num
[0]) { /* hiding the desktop, showing the clients */
127 if (!follow_mouse
|| !focus_under_pointer())
128 focus_fallback(TRUE
);
129 } else /* hiding clients, showing the desktop */
133 case Event_X_EnterNotify
:
135 if (e->data.x.client != NULL)
136 g_message("skipped enter %lx", e->data.x.client->window);
138 g_message("skipped enter 'root'");
142 if (e
->data
.x
.client
!= NULL
&& client_normal(e
->data
.x
.client
))
143 client_focus(e
->data
.x
.client
);
147 g_assert_not_reached();
151 void plugin_startup()
153 dispatch_register(Event_Client_Mapped
|
155 Event_Client_Unmapped
|
156 Event_X_EnterNotify
|
157 Event_Ob_ShowDesktop
,
158 (EventHandler
)event
, NULL
);
161 void plugin_shutdown()
163 dispatch_register(0, (EventHandler
)event
, NULL
);