2001-06-06 22:42:24 -04:00
|
|
|
/* Metacity window menu */
|
|
|
|
|
|
|
|
/*
|
|
|
|
* Copyright (C) 2001 Havoc Pennington
|
|
|
|
*
|
|
|
|
* This program is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU General Public License as
|
|
|
|
* published by the Free Software Foundation; either version 2 of the
|
|
|
|
* License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful, but
|
|
|
|
* WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU General Public License
|
|
|
|
* along with this program; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA
|
|
|
|
* 02111-1307, USA.
|
|
|
|
*/
|
|
|
|
|
2001-12-09 17:41:12 -05:00
|
|
|
#include <config.h>
|
2002-07-23 16:52:22 -04:00
|
|
|
#include <stdio.h>
|
2002-07-24 14:28:56 -04:00
|
|
|
#include <string.h>
|
2001-06-06 22:42:24 -04:00
|
|
|
#include "menu.h"
|
|
|
|
#include "main.h"
|
2001-06-21 02:08:35 -04:00
|
|
|
#include "util.h"
|
|
|
|
#include "core.h"
|
2002-02-06 22:07:56 -05:00
|
|
|
#include "themewidget.h"
|
2001-06-09 02:08:44 -04:00
|
|
|
|
2001-06-06 22:42:24 -04:00
|
|
|
typedef struct _MenuItem MenuItem;
|
|
|
|
typedef struct _MenuData MenuData;
|
|
|
|
|
|
|
|
struct _MenuItem
|
|
|
|
{
|
2001-06-20 23:40:14 -04:00
|
|
|
MetaMenuOp op;
|
2001-06-06 22:42:24 -04:00
|
|
|
const char *stock_id;
|
|
|
|
const char *label;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
struct _MenuData
|
|
|
|
{
|
2001-06-23 01:49:35 -04:00
|
|
|
MetaWindowMenu *menu;
|
2001-06-21 02:08:35 -04:00
|
|
|
MetaMenuOp op;
|
2001-06-06 22:42:24 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
static void activate_cb (GtkWidget *menuitem, gpointer data);
|
|
|
|
|
|
|
|
static MenuItem menuitems[] = {
|
2002-08-10 12:47:43 -04:00
|
|
|
{ META_MENU_OP_MINIMIZE, METACITY_STOCK_MINIMIZE, N_("_Minimize") },
|
|
|
|
{ META_MENU_OP_MAXIMIZE, METACITY_STOCK_MAXIMIZE, N_("Ma_ximize") },
|
2002-04-30 07:00:53 -04:00
|
|
|
{ META_MENU_OP_UNMAXIMIZE, NULL, N_("Unma_ximize") },
|
2001-06-21 02:08:35 -04:00
|
|
|
{ META_MENU_OP_SHADE, NULL, N_("_Shade") },
|
2002-04-30 07:00:53 -04:00
|
|
|
{ META_MENU_OP_UNSHADE, NULL, N_("Un_shade") },
|
2001-07-11 02:22:00 -04:00
|
|
|
{ META_MENU_OP_MOVE, NULL, N_("Mo_ve") },
|
|
|
|
{ META_MENU_OP_RESIZE, NULL, N_("_Resize") },
|
2001-06-09 23:17:15 -04:00
|
|
|
{ 0, NULL, NULL }, /* separator */
|
2002-08-10 14:47:36 -04:00
|
|
|
{ META_MENU_OP_DELETE, METACITY_STOCK_DELETE, N_("_Close") },
|
|
|
|
{ 0, NULL, NULL }, /* separator */
|
2001-06-21 02:08:35 -04:00
|
|
|
{ META_MENU_OP_STICK, NULL, N_("Put on _All Workspaces") },
|
|
|
|
{ META_MENU_OP_UNSTICK, NULL, N_("Only on _This Workspace") }
|
2001-06-06 22:42:24 -04:00
|
|
|
};
|
|
|
|
|
|
|
|
static void
|
|
|
|
popup_position_func (GtkMenu *menu,
|
|
|
|
gint *x,
|
|
|
|
gint *y,
|
|
|
|
gboolean *push_in,
|
|
|
|
gpointer user_data)
|
|
|
|
{
|
|
|
|
GtkRequisition req;
|
|
|
|
GdkPoint *pos;
|
|
|
|
|
|
|
|
pos = user_data;
|
|
|
|
|
|
|
|
gtk_widget_size_request (GTK_WIDGET (menu), &req);
|
|
|
|
|
|
|
|
*x = pos->x;
|
|
|
|
*y = pos->y;
|
|
|
|
|
|
|
|
/* Ensure onscreen */
|
|
|
|
*x = CLAMP (*x, 0, MAX (0, gdk_screen_width () - req.width));
|
|
|
|
*y = CLAMP (*y, 0, MAX (0, gdk_screen_height () - req.height));
|
|
|
|
}
|
|
|
|
|
2001-06-21 02:08:35 -04:00
|
|
|
static void
|
2001-06-23 01:49:35 -04:00
|
|
|
menu_closed (GtkMenu *widget,
|
2001-06-21 02:08:35 -04:00
|
|
|
gpointer data)
|
|
|
|
{
|
2001-06-23 01:49:35 -04:00
|
|
|
MetaWindowMenu *menu;
|
|
|
|
|
|
|
|
menu = data;
|
2001-06-09 23:17:15 -04:00
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
meta_frames_notify_menu_hide (menu->frames);
|
|
|
|
(* menu->func) (menu, gdk_display,
|
|
|
|
menu->client_xwindow,
|
|
|
|
0, 0,
|
|
|
|
menu->data);
|
|
|
|
|
|
|
|
/* menu may now be freed */
|
2001-06-09 01:14:43 -04:00
|
|
|
}
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
static void
|
|
|
|
activate_cb (GtkWidget *menuitem, gpointer data)
|
2001-06-06 22:42:24 -04:00
|
|
|
{
|
2001-06-23 01:49:35 -04:00
|
|
|
MenuData *md;
|
2001-06-06 22:42:24 -04:00
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
g_return_if_fail (GTK_IS_WIDGET (menuitem));
|
2001-06-21 02:08:35 -04:00
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
md = data;
|
2001-06-21 02:08:35 -04:00
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
meta_frames_notify_menu_hide (md->menu->frames);
|
|
|
|
(* md->menu->func) (md->menu, gdk_display,
|
|
|
|
md->menu->client_xwindow,
|
|
|
|
md->op,
|
|
|
|
GPOINTER_TO_INT (g_object_get_data (G_OBJECT (menuitem),
|
|
|
|
"workspace")),
|
|
|
|
md->menu->data);
|
2001-06-06 22:42:24 -04:00
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
/* menu may now be freed */
|
|
|
|
}
|
2001-06-06 22:42:24 -04:00
|
|
|
|
2002-07-23 16:52:22 -04:00
|
|
|
/*
|
|
|
|
* Given a Display and an index, get the workspace name and add any
|
|
|
|
* accelerators. At the moment this means adding a _ if the name is of
|
|
|
|
* the form "Workspace n" where n is less than 10, and escaping any
|
|
|
|
* other '_'s so they do not create inadvertant accelerators.
|
|
|
|
*
|
|
|
|
* The calling code owns the string, and is reponsible to free the
|
|
|
|
* memory after use.
|
|
|
|
*/
|
|
|
|
static char *
|
|
|
|
get_workspace_name_with_accel (Display *display,
|
|
|
|
int index)
|
|
|
|
{
|
|
|
|
char *name;
|
|
|
|
unsigned int number;
|
|
|
|
|
|
|
|
name = meta_core_get_workspace_name_with_index (display, index);
|
|
|
|
|
|
|
|
/*
|
|
|
|
* If the name is of the form "Workspace x" where x is an unsigned
|
|
|
|
* integer, insert a '_' before the number if it is less than 10 and
|
|
|
|
* return it
|
|
|
|
*/
|
|
|
|
if (sscanf (name, _("Workspace %u"), &number) == 1)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Above name is a pointer into the Workspace struct. Here we make
|
|
|
|
* a copy copy so we can have our wicked way with it.
|
|
|
|
*/
|
|
|
|
name = g_strdup_printf (_("Workspace %s%d"),
|
|
|
|
number < 10 ? "_" : "",
|
|
|
|
number);
|
|
|
|
return name;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
* Otherwise this is just a normal name to which we cannot really
|
|
|
|
* add accelerators. Escape any _ characters so that the user's
|
|
|
|
* workspace names do not get mangled.
|
|
|
|
*/
|
|
|
|
char *new_name, *source, *dest;
|
|
|
|
source = name;
|
|
|
|
/*
|
|
|
|
* Assume the worst case, that every character is a _
|
|
|
|
*/
|
|
|
|
dest = new_name = g_malloc0 (strlen (name) * 2);
|
|
|
|
/*
|
|
|
|
* Now iterate down the strings, adding '_' to escape as we go
|
|
|
|
*/
|
|
|
|
while (*source != '\0')
|
|
|
|
{
|
|
|
|
if (*source == '_')
|
|
|
|
*dest++ = '_';
|
|
|
|
*dest++ = *source++;
|
|
|
|
}
|
|
|
|
/*
|
|
|
|
* We don't free *name as we don't own it, and pass ownership of
|
|
|
|
* *new_name to the calling code.
|
|
|
|
*/
|
|
|
|
return new_name;
|
|
|
|
}
|
|
|
|
}
|
2002-02-06 22:07:56 -05:00
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
MetaWindowMenu*
|
|
|
|
meta_window_menu_new (MetaFrames *frames,
|
|
|
|
MetaMenuOp ops,
|
|
|
|
MetaMenuOp insensitive,
|
|
|
|
Window client_xwindow,
|
|
|
|
int active_workspace,
|
|
|
|
int n_workspaces,
|
|
|
|
MetaWindowMenuFunc func,
|
|
|
|
gpointer data)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
MetaWindowMenu *menu;
|
|
|
|
|
|
|
|
menu = g_new (MetaWindowMenu, 1);
|
|
|
|
menu->frames = frames;
|
|
|
|
menu->client_xwindow = client_xwindow;
|
|
|
|
menu->func = func;
|
|
|
|
menu->data = data;
|
|
|
|
menu->ops = ops;
|
|
|
|
menu->insensitive = insensitive;
|
2001-06-21 02:08:35 -04:00
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
menu->menu = gtk_menu_new ();
|
2002-05-08 12:24:28 -04:00
|
|
|
#ifdef HAVE_GTK_MULTIHEAD
|
|
|
|
gtk_menu_set_screen (GTK_MENU (menu->menu),
|
|
|
|
gtk_widget_get_screen (GTK_WIDGET (frames)));
|
|
|
|
#endif
|
2001-06-06 22:42:24 -04:00
|
|
|
i = 0;
|
2002-02-06 22:07:56 -05:00
|
|
|
while (i < (int) G_N_ELEMENTS (menuitems))
|
2001-06-06 22:42:24 -04:00
|
|
|
{
|
2001-06-09 23:17:15 -04:00
|
|
|
if (ops & menuitems[i].op || menuitems[i].op == 0)
|
2001-06-06 22:42:24 -04:00
|
|
|
{
|
|
|
|
GtkWidget *mi;
|
|
|
|
MenuData *md;
|
2001-06-09 23:17:15 -04:00
|
|
|
|
|
|
|
if (menuitems[i].op == 0)
|
2001-06-06 22:42:24 -04:00
|
|
|
{
|
2001-06-09 23:17:15 -04:00
|
|
|
mi = gtk_separator_menu_item_new ();
|
2001-06-06 22:42:24 -04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2001-06-21 02:08:35 -04:00
|
|
|
GtkWidget *image;
|
|
|
|
|
|
|
|
image = NULL;
|
|
|
|
|
2002-08-10 12:47:43 -04:00
|
|
|
if (menuitems[i].stock_id)
|
2001-06-21 02:08:35 -04:00
|
|
|
{
|
2001-06-09 23:17:15 -04:00
|
|
|
image = gtk_image_new_from_stock (menuitems[i].stock_id,
|
|
|
|
GTK_ICON_SIZE_MENU);
|
2001-06-21 02:08:35 -04:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (image)
|
|
|
|
{
|
2002-03-31 16:19:58 -05:00
|
|
|
mi = gtk_image_menu_item_new_with_mnemonic (_(menuitems[i].label));
|
2001-06-09 23:17:15 -04:00
|
|
|
gtk_image_menu_item_set_image (GTK_IMAGE_MENU_ITEM (mi),
|
|
|
|
image);
|
|
|
|
gtk_widget_show (image);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2002-03-31 16:19:58 -05:00
|
|
|
mi = gtk_menu_item_new_with_mnemonic (_(menuitems[i].label));
|
2001-06-09 23:17:15 -04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (insensitive & menuitems[i].op)
|
|
|
|
gtk_widget_set_sensitive (mi, FALSE);
|
|
|
|
|
|
|
|
md = g_new (MenuData, 1);
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
md->menu = menu;
|
2001-06-09 23:17:15 -04:00
|
|
|
md->op = menuitems[i].op;
|
|
|
|
|
2001-06-21 02:08:35 -04:00
|
|
|
gtk_signal_connect_full (GTK_OBJECT (mi),
|
|
|
|
"activate",
|
|
|
|
GTK_SIGNAL_FUNC (activate_cb),
|
|
|
|
NULL,
|
|
|
|
md,
|
|
|
|
g_free, FALSE, FALSE);
|
2001-06-06 22:42:24 -04:00
|
|
|
}
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu->menu),
|
2001-06-06 22:42:24 -04:00
|
|
|
mi);
|
|
|
|
|
|
|
|
gtk_widget_show (mi);
|
|
|
|
}
|
|
|
|
++i;
|
|
|
|
}
|
2001-06-09 01:14:43 -04:00
|
|
|
|
2001-06-21 02:08:35 -04:00
|
|
|
if (ops & META_MENU_OP_WORKSPACES)
|
2001-06-09 01:14:43 -04:00
|
|
|
{
|
2002-03-06 22:46:12 -05:00
|
|
|
meta_verbose ("Creating %d-workspace menu current space %d\n",
|
2001-06-23 01:49:35 -04:00
|
|
|
n_workspaces, active_workspace);
|
2001-06-09 01:14:43 -04:00
|
|
|
|
2001-06-09 23:17:15 -04:00
|
|
|
if (n_workspaces > 0)
|
2001-06-09 01:14:43 -04:00
|
|
|
{
|
2001-06-09 02:08:44 -04:00
|
|
|
GtkWidget *mi;
|
2002-07-23 16:52:22 -04:00
|
|
|
Display *display;
|
|
|
|
|
|
|
|
display = gdk_x11_drawable_get_xdisplay(GTK_WIDGET(frames)->window);
|
|
|
|
|
2001-06-09 01:14:43 -04:00
|
|
|
i = 0;
|
|
|
|
while (i < n_workspaces)
|
|
|
|
{
|
2002-07-23 16:52:22 -04:00
|
|
|
char *label, *name;
|
2001-06-09 01:14:43 -04:00
|
|
|
MenuData *md;
|
2001-06-09 23:17:15 -04:00
|
|
|
|
2002-07-23 16:52:22 -04:00
|
|
|
name = get_workspace_name_with_accel (display, i);
|
2001-06-23 01:49:35 -04:00
|
|
|
if (ops & META_MENU_OP_UNSTICK)
|
2002-07-23 16:52:22 -04:00
|
|
|
label = g_strdup_printf (_("Only on %s"), name);
|
2001-06-09 23:17:15 -04:00
|
|
|
else
|
2002-07-23 16:52:22 -04:00
|
|
|
label = g_strdup_printf(_("Move to %s"), name);
|
2001-06-09 01:14:43 -04:00
|
|
|
mi = gtk_menu_item_new_with_mnemonic (label);
|
|
|
|
|
2002-07-23 16:52:22 -04:00
|
|
|
g_free (name);
|
2001-06-09 01:14:43 -04:00
|
|
|
g_free (label);
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
if (!(ops & META_MENU_OP_UNSTICK) &&
|
|
|
|
(active_workspace == i ||
|
2001-06-21 02:08:35 -04:00
|
|
|
insensitive & META_MENU_OP_WORKSPACES))
|
2001-06-09 01:14:43 -04:00
|
|
|
gtk_widget_set_sensitive (mi, FALSE);
|
2002-05-26 11:54:38 -04:00
|
|
|
else if (insensitive & META_MENU_OP_WORKSPACES)
|
|
|
|
gtk_widget_set_sensitive (mi, FALSE);
|
2001-06-09 01:14:43 -04:00
|
|
|
md = g_new (MenuData, 1);
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
md->menu = menu;
|
2001-06-21 02:08:35 -04:00
|
|
|
md->op = META_MENU_OP_WORKSPACES;
|
2001-06-09 01:14:43 -04:00
|
|
|
|
|
|
|
g_object_set_data (G_OBJECT (mi),
|
|
|
|
"workspace",
|
|
|
|
GINT_TO_POINTER (i));
|
|
|
|
|
2001-06-21 02:08:35 -04:00
|
|
|
gtk_signal_connect_full (GTK_OBJECT (mi),
|
|
|
|
"activate",
|
|
|
|
GTK_SIGNAL_FUNC (activate_cb),
|
|
|
|
NULL,
|
|
|
|
md,
|
|
|
|
g_free, FALSE, FALSE);
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
gtk_menu_shell_append (GTK_MENU_SHELL (menu->menu),
|
2001-06-09 01:14:43 -04:00
|
|
|
mi);
|
|
|
|
|
|
|
|
gtk_widget_show (mi);
|
|
|
|
|
|
|
|
++i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
2001-06-21 02:08:35 -04:00
|
|
|
meta_verbose ("not creating workspace menu\n");
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
gtk_signal_connect (GTK_OBJECT (menu->menu),
|
2001-06-21 02:08:35 -04:00
|
|
|
"selection_done",
|
|
|
|
GTK_SIGNAL_FUNC (menu_closed),
|
2001-06-23 01:49:35 -04:00
|
|
|
menu);
|
|
|
|
|
|
|
|
return menu;
|
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
meta_window_menu_popup (MetaWindowMenu *menu,
|
|
|
|
int root_x,
|
|
|
|
int root_y,
|
|
|
|
int button,
|
|
|
|
guint32 timestamp)
|
|
|
|
{
|
|
|
|
GdkPoint *pt;
|
2001-06-06 22:42:24 -04:00
|
|
|
|
|
|
|
pt = g_new (GdkPoint, 1);
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
g_object_set_data_full (G_OBJECT (menu->menu),
|
2001-06-06 22:42:24 -04:00
|
|
|
"destroy-point",
|
|
|
|
pt,
|
|
|
|
g_free);
|
|
|
|
|
|
|
|
pt->x = root_x;
|
|
|
|
pt->y = root_y;
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
gtk_menu_popup (GTK_MENU (menu->menu),
|
2001-06-06 22:42:24 -04:00
|
|
|
NULL, NULL,
|
|
|
|
popup_position_func, pt,
|
|
|
|
button,
|
|
|
|
timestamp);
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
if (!GTK_MENU_SHELL (menu->menu)->have_xgrab)
|
2001-06-21 02:08:35 -04:00
|
|
|
meta_warning ("GtkMenu failed to grab the pointer\n");
|
2001-06-09 01:14:43 -04:00
|
|
|
}
|
|
|
|
|
2001-06-23 01:49:35 -04:00
|
|
|
void
|
|
|
|
meta_window_menu_free (MetaWindowMenu *menu)
|
2001-06-06 22:42:24 -04:00
|
|
|
{
|
2001-06-23 01:49:35 -04:00
|
|
|
gtk_widget_destroy (menu->menu);
|
|
|
|
g_free (menu);
|
2001-06-06 22:42:24 -04:00
|
|
|
}
|