1 /* -*- indent-tabs-mode: nil; tab-width: 4; c-basic-offset: 4; -*-
3 client_menu.c for the Openbox window manager
4 Copyright (c) 2003-2007 Dana Jansens
6 This program is free software; you can redistribute it and/or modify
7 it under the terms of the GNU General Public License as published by
8 the Free Software Foundation; either version 2 of the License, or
9 (at your option) any later version.
11 This program is distributed in the hope that it will be useful,
12 but WITHOUT ANY WARRANTY; without even the implied warranty of
13 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
14 GNU General Public License for more details.
16 See the COPYING file for a copy of the GNU General Public License.
21 #include "menuframe.h"
25 #include "client_menu.h"
28 #include "moveresize.h"
35 #define CLIENT_MENU_NAME "client-menu"
36 #define SEND_TO_MENU_NAME "client-send-to-menu"
37 #define LAYER_MENU_NAME "client-layer-menu"
58 static void set_icon_color(ObMenuEntry
*e
)
60 e
->data
.normal
.mask_normal_color
= ob_rr_theme
->menu_color
;
61 e
->data
.normal
.mask_selected_color
= ob_rr_theme
->menu_selected_color
;
62 e
->data
.normal
.mask_disabled_color
= ob_rr_theme
->menu_disabled_color
;
63 e
->data
.normal
.mask_disabled_selected_color
=
64 ob_rr_theme
->menu_disabled_selected_color
;
67 static gboolean
client_menu_update(ObMenuFrame
*frame
, gpointer data
)
69 ObMenu
*menu
= frame
->menu
;
72 if (frame
->client
== NULL
|| !client_normal(frame
->client
))
73 return FALSE
; /* don't show the menu */
75 for (it
= menu
->entries
; it
; it
= g_list_next(it
)) {
76 ObMenuEntry
*e
= it
->data
;
77 gboolean
*en
= &e
->data
.normal
.enabled
; /* save some typing */
78 ObClient
*c
= frame
->client
;
80 if (e
->type
== OB_MENU_ENTRY_TYPE_NORMAL
) {
83 *en
= c
->functions
& OB_CLIENT_FUNC_ICONIFY
;
86 *en
= c
->max_horz
|| c
->max_vert
;
89 *en
= ((c
->functions
& OB_CLIENT_FUNC_MAXIMIZE
) &&
90 (!c
->max_horz
|| !c
->max_vert
));
93 *en
= c
->functions
& OB_CLIENT_FUNC_SHADE
;
96 *en
= c
->functions
& OB_CLIENT_FUNC_MOVE
;
99 *en
= c
->functions
& OB_CLIENT_FUNC_RESIZE
;
102 *en
= c
->functions
& OB_CLIENT_FUNC_CLOSE
;
104 case CLIENT_DECORATE
:
105 *en
= c
->functions
& OB_CLIENT_FUNC_UNDECORATE
;
112 return TRUE
; /* show the menu */
115 static void client_menu_execute(ObMenuEntry
*e
, ObMenuFrame
*f
,
116 ObClient
*c
, guint state
, gpointer data
)
123 if (!config_focus_under_mouse
)
124 ignore_start
= event_start_ignore_all_enters();
128 /* the client won't be on screen anymore so hide the menu */
129 menu_frame_hide_all();
130 f
= NULL
; /* and don't update */
132 client_iconify(c
, TRUE
, FALSE
, FALSE
);
135 client_maximize(c
, FALSE
, 0);
137 case CLIENT_MAXIMIZE
:
138 client_maximize(c
, TRUE
, 0);
141 client_shade(c
, !c
->shaded
);
143 case CLIENT_DECORATE
:
144 client_set_undecorated(c
, !c
->undecorated
);
147 /* this needs to grab the keyboard so hide the menu */
148 menu_frame_hide_all();
149 f
= NULL
; /* and don't update */
151 screen_pointer_pos(&x
, &y
);
152 moveresize_start(c
, x
, y
, 0,
153 prop_atoms
.net_wm_moveresize_move_keyboard
);
156 /* this needs to grab the keyboard so hide the menu */
157 menu_frame_hide_all();
158 f
= NULL
; /* and don't update */
160 screen_pointer_pos(&x
, &y
);
161 moveresize_start(c
, x
, y
, 0,
162 prop_atoms
.net_wm_moveresize_size_keyboard
);
168 g_assert_not_reached();
171 if (!config_focus_under_mouse
)
172 event_end_ignore_all_enters(ignore_start
);
174 /* update the menu cuz stuff can have changed */
176 client_menu_update(f
, NULL
);
177 menu_frame_render(f
);
181 static gboolean
layer_menu_update(ObMenuFrame
*frame
, gpointer data
)
183 ObMenu
*menu
= frame
->menu
;
186 if (frame
->client
== NULL
|| !client_normal(frame
->client
))
187 return FALSE
; /* don't show the menu */
189 for (it
= menu
->entries
; it
; it
= g_list_next(it
)) {
190 ObMenuEntry
*e
= it
->data
;
191 gboolean
*en
= &e
->data
.normal
.enabled
; /* save some typing */
192 ObClient
*c
= frame
->client
;
194 if (e
->type
== OB_MENU_ENTRY_TYPE_NORMAL
) {
197 *en
= !c
->above
&& (c
->functions
& OB_CLIENT_FUNC_ABOVE
);
200 *en
= c
->above
|| c
->below
;
203 *en
= !c
->below
&& (c
->functions
& OB_CLIENT_FUNC_BELOW
);
210 return TRUE
; /* show the menu */
213 static void layer_menu_execute(ObMenuEntry
*e
, ObMenuFrame
*f
,
214 ObClient
*c
, guint state
, gpointer data
)
220 if (!config_focus_under_mouse
)
221 ignore_start
= event_start_ignore_all_enters();
223 client_set_layer(c
, e
->id
);
225 if (!config_focus_under_mouse
)
226 event_end_ignore_all_enters(ignore_start
);
228 /* update the menu cuz stuff can have changed */
230 layer_menu_update(f
, NULL
);
231 menu_frame_render(f
);
235 static gboolean
send_to_menu_update(ObMenuFrame
*frame
, gpointer data
)
237 ObMenu
*menu
= frame
->menu
;
238 ObClient
*c
= frame
->client
;
243 if (c
== NULL
|| !client_normal(c
))
244 return FALSE
; /* don't show the menu */
247 menu_clear_entries(menu
);
249 if (!menu
->entries
) {
250 for (i
= 0; i
<= screen_num_desktops
; ++i
) {
254 if (i
== screen_num_desktops
) {
255 menu_add_separator(menu
, -1, NULL
);
258 name
= _("All desktops");
261 name
= screen_desktop_names
[i
];
264 e
= menu_add_normal(menu
, desk
, name
, NULL
, FALSE
);
269 for (it
= menu
->entries
; it
; it
= g_list_next(it
)) {
270 ObMenuEntry
*e
= it
->data
;
273 e
->data
.normal
.enabled
= c
->desktop
!= desk
;
275 if ((desk
== DESKTOP_ALL
&& c
->desktop
!= DESKTOP_ALL
) ||
276 (c
->desktop
== DESKTOP_ALL
&& desk
== screen_desktop
))
278 e
->data
.normal
.mask
= ob_rr_theme
->desk_mask
;
281 e
->data
.normal
.mask
= NULL
;
284 return TRUE
; /* show the menu */
287 static void send_to_menu_execute(ObMenuEntry
*e
, ObMenuFrame
*f
,
288 ObClient
*c
, guint state
, gpointer data
)
292 client_set_desktop(c
, e
->id
, FALSE
, FALSE
);
293 if (f
&& c
->desktop
!= screen_desktop
&& c
->desktop
!= DESKTOP_ALL
)
294 /* the client won't even be on the screen anymore, so hide the menu */
295 menu_frame_hide_all();
297 send_to_menu_update(f
, (gpointer
)1);
298 menu_frame_render(f
);
302 static void client_menu_place(ObMenuFrame
*frame
, gint
*x
, gint
*y
,
303 gboolean mouse
, gpointer data
)
307 if (!mouse
&& frame
->client
) {
308 *x
= frame
->client
->frame
->area
.x
;
310 /* try below the titlebar */
311 *y
= frame
->client
->frame
->area
.y
+ frame
->client
->frame
->size
.top
-
312 frame
->client
->frame
->bwidth
;
313 menu_frame_move_on_screen(frame
, *x
, *y
, &dx
, &dy
);
315 /* try above the titlebar */
316 *y
= frame
->client
->frame
->area
.y
+ frame
->client
->frame
->bwidth
-
318 menu_frame_move_on_screen(frame
, *x
, *y
, &dx
, &dy
);
321 /* didnt fit either way, use move on screen's values */
322 *y
= frame
->client
->frame
->area
.y
+ frame
->client
->frame
->size
.top
;
323 menu_frame_move_on_screen(frame
, *x
, *y
, &dx
, &dy
);
334 /* try to the bottom right of the cursor */
335 menu_frame_move_on_screen(frame
, myx
, myy
, &dx
, &dy
);
336 if (dx
!= 0 || dy
!= 0) {
337 /* try to the bottom left of the cursor */
338 myx
= *x
- frame
->area
.width
;
340 menu_frame_move_on_screen(frame
, myx
, myy
, &dx
, &dy
);
342 if (dx
!= 0 || dy
!= 0) {
343 /* try to the top right of the cursor */
345 myy
= *y
- frame
->area
.height
;
346 menu_frame_move_on_screen(frame
, myx
, myy
, &dx
, &dy
);
348 if (dx
!= 0 || dy
!= 0) {
349 /* try to the top left of the cursor */
350 myx
= *x
- frame
->area
.width
;
351 myy
= *y
- frame
->area
.height
;
352 menu_frame_move_on_screen(frame
, myx
, myy
, &dx
, &dy
);
354 if (dx
!= 0 || dy
!= 0) {
355 /* if didnt fit on either side so just use what it says */
358 menu_frame_move_on_screen(frame
, myx
, myy
, &dx
, &dy
);
365 void client_menu_startup(void)
370 menu
= menu_new(LAYER_MENU_NAME
, _("_Layer"), TRUE
, NULL
);
371 menu_show_all_shortcuts(menu
, TRUE
);
372 menu_set_update_func(menu
, layer_menu_update
);
373 menu_set_execute_func(menu
, layer_menu_execute
);
375 menu_add_normal(menu
, LAYER_TOP
, _("Always on _top"), NULL
, TRUE
);
376 menu_add_normal(menu
, LAYER_NORMAL
, _("_Normal"), NULL
, TRUE
);
377 menu_add_normal(menu
, LAYER_BOTTOM
, _("Always on _bottom"),NULL
, TRUE
);
379 menu
= menu_new(SEND_TO_MENU_NAME
, _("_Send to desktop"), TRUE
, NULL
);
380 menu_set_update_func(menu
, send_to_menu_update
);
381 menu_set_execute_func(menu
, send_to_menu_execute
);
383 menu
= menu_new(CLIENT_MENU_NAME
, _("Client menu"), TRUE
, NULL
);
384 menu_show_all_shortcuts(menu
, TRUE
);
385 menu_set_update_func(menu
, client_menu_update
);
386 menu_set_place_func(menu
, client_menu_place
);
387 menu_set_execute_func(menu
, client_menu_execute
);
389 menu_add_submenu(menu
, CLIENT_SEND_TO
, SEND_TO_MENU_NAME
);
391 menu_add_submenu(menu
, CLIENT_LAYER
, LAYER_MENU_NAME
);
393 e
= menu_add_normal(menu
, CLIENT_RESTORE
, _("R_estore"), NULL
, TRUE
);
394 e
->data
.normal
.mask
= ob_rr_theme
->max_toggled_mask
;
397 menu_add_normal(menu
, CLIENT_MOVE
, _("_Move"), NULL
, TRUE
);
399 menu_add_normal(menu
, CLIENT_RESIZE
, _("Resi_ze"), NULL
, TRUE
);
401 e
= menu_add_normal(menu
, CLIENT_ICONIFY
, _("Ico_nify"), NULL
, TRUE
);
402 e
->data
.normal
.mask
= ob_rr_theme
->iconify_mask
;
405 e
= menu_add_normal(menu
, CLIENT_MAXIMIZE
, _("Ma_ximize"), NULL
, TRUE
);
406 e
->data
.normal
.mask
= ob_rr_theme
->max_mask
;
409 menu_add_normal(menu
, CLIENT_SHADE
, _("_Roll up/down"), NULL
, TRUE
);
411 menu_add_normal(menu
, CLIENT_DECORATE
, _("Un/_Decorate"), NULL
, TRUE
);
413 menu_add_separator(menu
, -1, NULL
);
415 e
= menu_add_normal(menu
, CLIENT_CLOSE
, _("_Close"), NULL
, TRUE
);
416 e
->data
.normal
.mask
= ob_rr_theme
->close_mask
;