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"
26 #include "moveresize.h"
32 #define CLIENT_MENU_NAME "client-menu"
33 #define SEND_TO_MENU_NAME "client-send-to-menu"
34 #define LAYER_MENU_NAME "client-layer-menu"
55 static gboolean
client_menu_update(ObMenuFrame
*frame
, gpointer data
)
57 ObMenu
*menu
= frame
->menu
;
60 if (frame
->client
== NULL
|| !client_normal(frame
->client
))
61 return FALSE
; /* don't show the menu */
63 for (it
= menu
->entries
; it
; it
= g_list_next(it
)) {
64 ObMenuEntry
*e
= it
->data
;
65 gboolean
*en
= &e
->data
.normal
.enabled
; /* save some typing */
66 ObClient
*c
= frame
->client
;
68 if (e
->type
== OB_MENU_ENTRY_TYPE_NORMAL
) {
71 *en
= c
->functions
& OB_CLIENT_FUNC_ICONIFY
;
74 *en
= c
->max_horz
|| c
->max_vert
;
77 *en
= ((c
->functions
& OB_CLIENT_FUNC_MAXIMIZE
) &&
78 (!c
->max_horz
|| !c
->max_vert
));
81 *en
= c
->functions
& OB_CLIENT_FUNC_SHADE
;
84 *en
= c
->functions
& OB_CLIENT_FUNC_MOVE
;
87 *en
= c
->functions
& OB_CLIENT_FUNC_RESIZE
;
90 *en
= c
->functions
& OB_CLIENT_FUNC_CLOSE
;
93 *en
= client_normal(c
);
100 return TRUE
; /* show the menu */
103 static void client_menu_execute(ObMenuEntry
*e
, guint state
, gpointer data
,
111 for (it
= menu_frame_visible
; it
; it
= g_list_next(it
)) {
113 /* yay this is our menu frame */
114 if (f
->menu
== e
->menu
)
123 client_iconify(c
, TRUE
, FALSE
);
126 client_maximize(c
, FALSE
, 0);
128 case CLIENT_MAXIMIZE
:
129 client_maximize(c
, TRUE
, 0);
132 client_shade(c
, !c
->shaded
);
134 case CLIENT_DECORATE
:
135 client_set_undecorated(c
, !c
->undecorated
);
138 moveresize_start(c
,0,0,0, prop_atoms
.net_wm_moveresize_move_keyboard
);
141 moveresize_start(c
,0,0,0,prop_atoms
.net_wm_moveresize_size_keyboard
);
147 g_assert_not_reached();
150 /* update the menu cuz stuff can have changed */
151 client_menu_update(f
, NULL
);
152 menu_frame_render(f
);
155 static gboolean
layer_menu_update(ObMenuFrame
*frame
, gpointer data
)
157 ObMenu
*menu
= frame
->menu
;
160 if (frame
->client
== NULL
|| !client_normal(frame
->client
))
161 return FALSE
; /* don't show the menu */
163 for (it
= menu
->entries
; it
; it
= g_list_next(it
)) {
164 ObMenuEntry
*e
= it
->data
;
165 gboolean
*en
= &e
->data
.normal
.enabled
; /* save some typing */
166 ObClient
*c
= frame
->client
;
168 if (e
->type
== OB_MENU_ENTRY_TYPE_NORMAL
) {
174 *en
= c
->above
|| c
->below
;
184 return TRUE
; /* show the menu */
187 static void layer_menu_execute(ObMenuEntry
*e
, guint state
, gpointer data
,
194 for (it
= menu_frame_visible
; it
; it
= g_list_next(it
)) {
196 /* yay this is our menu frame */
197 if (f
->menu
== e
->menu
)
204 client_set_layer(f
->client
, 1);
207 client_set_layer(f
->client
, 0);
210 client_set_layer(f
->client
, -1);
213 g_assert_not_reached();
216 /* update the menu cuz stuff can have changed */
217 layer_menu_update(f
, NULL
);
218 menu_frame_render(f
);
221 static gboolean
send_to_menu_update(ObMenuFrame
*frame
, gpointer data
)
223 ObMenu
*menu
= frame
->menu
;
227 menu_clear_entries(menu
);
229 if (frame
->client
== NULL
|| !client_normal(frame
->client
))
230 return FALSE
; /* don't show the menu */
232 for (i
= 0; i
<= screen_num_desktops
; ++i
) {
236 if (i
>= screen_num_desktops
) {
237 menu_add_separator(menu
, -1, NULL
);
240 name
= _("All desktops");
243 name
= screen_desktop_names
[i
];
246 e
= menu_add_normal(menu
, desk
, name
, NULL
, FALSE
);
248 if (desk
== DESKTOP_ALL
) {
249 e
->data
.normal
.mask
= ob_rr_theme
->desk_mask
;
250 e
->data
.normal
.mask_normal_color
= ob_rr_theme
->menu_color
;
251 e
->data
.normal
.mask_selected_color
=
252 ob_rr_theme
->menu_selected_color
;
253 e
->data
.normal
.mask_disabled_color
=
254 ob_rr_theme
->menu_disabled_color
;
255 e
->data
.normal
.mask_disabled_selected_color
=
256 ob_rr_theme
->menu_disabled_selected_color
;
259 if (frame
->client
->desktop
== desk
)
260 e
->data
.normal
.enabled
= FALSE
;
262 return TRUE
; /* show the menu */
265 static void send_to_menu_execute(ObMenuEntry
*e
, guint state
, gpointer data
,
272 for (it
= menu_frame_visible
; it
; it
= g_list_next(it
)) {
274 /* yay this is our menu frame */
275 if (f
->menu
== e
->menu
)
280 client_set_desktop(f
->client
, e
->id
, FALSE
);
281 /* the client won't even be on the screen anymore, so hide the menu */
282 menu_frame_hide_all();
285 static void client_menu_place(ObMenuFrame
*frame
, gint
*x
, gint
*y
,
286 gint button
, gpointer data
)
290 if (button
== 0 && frame
->client
) {
291 *x
= frame
->client
->frame
->area
.x
;
293 /* try below the titlebar */
294 *y
= frame
->client
->frame
->area
.y
+ frame
->client
->frame
->size
.top
-
295 frame
->client
->frame
->bwidth
;
296 menu_frame_move_on_screen(frame
, *x
, *y
, &dx
, &dy
);
298 /* try above the titlebar */
299 *y
= frame
->client
->frame
->area
.y
+ frame
->client
->frame
->bwidth
-
301 menu_frame_move_on_screen(frame
, *x
, *y
, &dx
, &dy
);
304 /* didnt fit either way, use move on screen's values */
305 *y
= frame
->client
->frame
->area
.y
+ frame
->client
->frame
->size
.top
;
306 menu_frame_move_on_screen(frame
, *x
, *y
, &dx
, &dy
);
317 /* try to the bottom right of the cursor */
318 menu_frame_move_on_screen(frame
, myx
, myy
, &dx
, &dy
);
319 if (dx
!= 0 || dy
!= 0) {
320 /* try to the bottom left of the cursor */
321 myx
= *x
- frame
->area
.width
;
323 menu_frame_move_on_screen(frame
, myx
, myy
, &dx
, &dy
);
325 if (dx
!= 0 || dy
!= 0) {
326 /* try to the top right of the cursor */
328 myy
= *y
- frame
->area
.height
;
329 menu_frame_move_on_screen(frame
, myx
, myy
, &dx
, &dy
);
331 if (dx
!= 0 || dy
!= 0) {
332 /* try to the top left of the cursor */
333 myx
= *x
- frame
->area
.width
;
334 myy
= *y
- frame
->area
.height
;
335 menu_frame_move_on_screen(frame
, myx
, myy
, &dx
, &dy
);
337 if (dx
!= 0 || dy
!= 0) {
338 /* if didnt fit on either side so just use what it says */
341 menu_frame_move_on_screen(frame
, myx
, myy
, &dx
, &dy
);
348 void client_menu_startup()
353 menu
= menu_new(LAYER_MENU_NAME
, _("&Layer"), TRUE
, NULL
);
354 menu_show_all_shortcuts(menu
, TRUE
);
355 menu_set_update_func(menu
, layer_menu_update
);
356 menu_set_execute_func(menu
, layer_menu_execute
);
358 menu_add_normal(menu
, LAYER_TOP
, _("Always on &top"), NULL
, TRUE
);
359 menu_add_normal(menu
, LAYER_NORMAL
, _("&Normal"), NULL
, TRUE
);
360 menu_add_normal(menu
, LAYER_BOTTOM
, _("Always on &bottom"),NULL
, TRUE
);
363 menu
= menu_new(SEND_TO_MENU_NAME
, _("&Send to desktop"), TRUE
, NULL
);
364 menu_set_update_func(menu
, send_to_menu_update
);
365 menu_set_execute_func(menu
, send_to_menu_execute
);
367 menu
= menu_new(CLIENT_MENU_NAME
, _("Client menu"), TRUE
, NULL
);
368 menu_show_all_shortcuts(menu
, TRUE
);
369 menu_set_update_func(menu
, client_menu_update
);
370 menu_set_place_func(menu
, client_menu_place
);
371 menu_set_execute_func(menu
, client_menu_execute
);
373 e
= menu_add_normal(menu
, CLIENT_RESTORE
, _("R&estore"), NULL
, TRUE
);
374 e
->data
.normal
.mask
= ob_rr_theme
->max_toggled_mask
;
375 e
->data
.normal
.mask_normal_color
= ob_rr_theme
->menu_color
;
376 e
->data
.normal
.mask_selected_color
= ob_rr_theme
->menu_selected_color
;
377 e
->data
.normal
.mask_disabled_color
= ob_rr_theme
->menu_disabled_color
;
378 e
->data
.normal
.mask_disabled_selected_color
=
379 ob_rr_theme
->menu_disabled_selected_color
;
381 menu_add_normal(menu
, CLIENT_MOVE
, _("&Move"), NULL
, TRUE
);
383 menu_add_normal(menu
, CLIENT_RESIZE
, _("Resi&ze"), NULL
, TRUE
);
385 e
= menu_add_normal(menu
, CLIENT_ICONIFY
, _("Ico&nify"), NULL
, TRUE
);
386 e
->data
.normal
.mask
= ob_rr_theme
->iconify_mask
;
387 e
->data
.normal
.mask_normal_color
= ob_rr_theme
->menu_color
;
388 e
->data
.normal
.mask_selected_color
= ob_rr_theme
->menu_selected_color
;
389 e
->data
.normal
.mask_disabled_color
= ob_rr_theme
->menu_disabled_color
;
390 e
->data
.normal
.mask_disabled_selected_color
=
391 ob_rr_theme
->menu_disabled_selected_color
;
393 e
= menu_add_normal(menu
, CLIENT_MAXIMIZE
, _("Ma&ximize"), NULL
, TRUE
);
394 e
->data
.normal
.mask
= ob_rr_theme
->max_mask
;
395 e
->data
.normal
.mask_normal_color
= ob_rr_theme
->menu_color
;
396 e
->data
.normal
.mask_selected_color
= ob_rr_theme
->menu_selected_color
;
397 e
->data
.normal
.mask_disabled_color
= ob_rr_theme
->menu_disabled_color
;
398 e
->data
.normal
.mask_disabled_selected_color
=
399 ob_rr_theme
->menu_disabled_selected_color
;
401 e
= menu_add_normal(menu
, CLIENT_SHADE
, _("&Roll up/down"), NULL
, TRUE
);
402 e
->data
.normal
.mask
= ob_rr_theme
->shade_mask
;
403 e
->data
.normal
.mask_normal_color
= ob_rr_theme
->menu_color
;
404 e
->data
.normal
.mask_selected_color
= ob_rr_theme
->menu_selected_color
;
405 e
->data
.normal
.mask_disabled_color
= ob_rr_theme
->menu_disabled_color
;
406 e
->data
.normal
.mask_disabled_selected_color
=
407 ob_rr_theme
->menu_disabled_selected_color
;
409 menu_add_normal(menu
, CLIENT_DECORATE
, _("Un/&Decorate"), NULL
, TRUE
);
411 menu_add_separator(menu
, -1, NULL
);
413 menu_add_submenu(menu
, CLIENT_SEND_TO
, SEND_TO_MENU_NAME
);
415 menu_add_submenu(menu
, CLIENT_LAYER
, LAYER_MENU_NAME
);
417 menu_add_separator(menu
, -1, NULL
);
419 e
= menu_add_normal(menu
, CLIENT_CLOSE
, _("&Close"), NULL
, TRUE
);
420 e
->data
.normal
.mask
= ob_rr_theme
->close_mask
;
421 e
->data
.normal
.mask_normal_color
= ob_rr_theme
->menu_color
;
422 e
->data
.normal
.mask_selected_color
= ob_rr_theme
->menu_selected_color
;
423 e
->data
.normal
.mask_disabled_color
= ob_rr_theme
->menu_disabled_color
;
424 e
->data
.normal
.mask_disabled_selected_color
=
425 ob_rr_theme
->menu_disabled_selected_color
;