/*
* Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
- * Copyright (C) 1999-2004 Hiroyuki Yamamoto
+ * Copyright (C) 1999-2013 Hiroyuki Yamamoto and the Claws Mail team
*
* 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
+ * the Free Software Foundation; either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* 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.
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ *
*/
#ifdef HAVE_CONFIG_H
# include "config.h"
+#include "claws-features.h"
#endif
#include <glib.h>
+#include <glib/gi18n.h>
#include <gtk/gtk.h>
-#include <gtk/gtkwidget.h>
-#include <gtk/gtkmenu.h>
-#include <gtk/gtkmenubar.h>
-#include <gtk/gtkcheckmenuitem.h>
-#include <gtk/gtkitemfactory.h>
-#include <gtk/gtkbutton.h>
-#include <gtk/gtkwindow.h>
-
-#include "intl.h"
+
+#include "gtkcmoptionmenu.h"
#include "menu.h"
#include "utils.h"
+#include "gtkutils.h"
+#include "defs.h"
-static void menu_item_add_accel( GtkWidget *widget, guint accel_signal_id, GtkAccelGroup *accel_group,
- guint accel_key, GdkModifierType accel_mods, GtkAccelFlags accel_flags,
- gpointer user_data);
+GtkActionGroup *cm_menu_create_action_group(const gchar *name, GtkActionEntry *entries,
+ gint num_entries, gpointer data)
+{
+ GtkActionGroup *group = gtk_action_group_new(name);
+ gtk_action_group_set_translate_func(group, menu_translate, NULL, NULL);
+ gtk_action_group_add_actions(group, entries, num_entries, data);
+ gtk_ui_manager_insert_action_group(gtkut_ui_manager(), group, 0);
+ return group;
+}
-static void menu_item_remove_accel(GtkWidget *widget, GtkAccelGroup *accel_group,
- guint accel_key, GdkModifierType accel_mods,
- gpointer user_data);
+GtkActionGroup *cm_menu_create_action_group_full(GtkUIManager *manager, const gchar *name, GtkActionEntry *entries,
+ gint num_entries, gpointer data)
+{
+ GtkActionGroup *group = gtk_action_group_new(name);
+ gtk_action_group_set_translate_func(group, menu_translate, NULL, NULL);
+ gtk_action_group_add_actions(group, entries, num_entries, data);
+ gtk_ui_manager_insert_action_group(manager, group, 0);
+ return group;
+}
-static void connect_accel_change_signals(GtkWidget* widget, GtkWidget *wid2) ;
+gchar *menu_translate(const gchar *path, gpointer data)
+{
+ gchar *retval;
+ retval = gettext(path);
-GtkWidget *menubar_create(GtkWidget *window, GtkItemFactoryEntry *entries,
- guint n_entries, const gchar *path, gpointer data)
-{
- GtkItemFactory *factory;
+ return retval;
+}
- factory = gtk_item_factory_new(GTK_TYPE_MENU_BAR, path, NULL);
- gtk_item_factory_set_translate_func(factory, menu_translate,
- NULL, NULL);
- gtk_item_factory_create_items(factory, n_entries, entries, data);
- gtk_window_add_accel_group (GTK_WINDOW (window), factory->accel_group);
+void cm_menu_set_sensitive(gchar *menu, gboolean sensitive)
+{
+ GtkUIManager *gui_manager = gtkut_ui_manager();
+ gchar *path = g_strdup_printf("Menus/%s", menu);
- return gtk_item_factory_get_widget(factory, path);
+ cm_menu_set_sensitive_full(gui_manager, path, sensitive);
+ g_free(path);
}
-GtkWidget *menu_create_items(GtkItemFactoryEntry *entries,
- guint n_entries, const gchar *path,
- GtkItemFactory **factory, gpointer data)
+void cm_toggle_menu_set_active(gchar *menu, gboolean active)
{
- *factory = gtk_item_factory_new(GTK_TYPE_MENU, path, NULL);
- gtk_item_factory_set_translate_func(*factory, menu_translate,
- NULL, NULL);
- gtk_item_factory_create_items(*factory, n_entries, entries, data);
+ GtkUIManager *gui_manager = gtkut_ui_manager();
+ gchar *path = g_strdup_printf("Menus/%s", menu);
- return gtk_item_factory_get_widget(*factory, path);
+ cm_toggle_menu_set_active_full(gui_manager, path, active);
+ g_free(path);
}
-GtkWidget *popupmenu_create(GtkWidget *window, GtkItemFactoryEntry *entries,
- guint n_entries, const gchar *path, gpointer data)
+void cm_menu_set_sensitive_full(GtkUIManager *gui_manager, const gchar *menu, gboolean sensitive)
{
- GtkItemFactory *factory;
- GtkAccelGroup *accel_group;
+ GtkWidget *widget;
+ gchar *path = g_strdup_printf("/%s/", menu);
+
+ widget = gtk_ui_manager_get_widget(gui_manager, path);
+ if( !GTK_IS_WIDGET(widget) ) {
+ g_message("Blah, '%s' is not a widget.\n", path);
+ }
- accel_group = gtk_accel_group_new();
- factory = gtk_item_factory_new(GTK_TYPE_MENU, path, accel_group);
- gtk_item_factory_set_translate_func(factory, menu_translate,
- NULL, NULL);
- gtk_item_factory_create_items(factory, n_entries, entries, data);
- gtk_window_add_accel_group(GTK_WINDOW (window), accel_group);
+ if( !GTK_IS_MENU_ITEM(widget) ) {
+ g_message("Blah, '%s' is not a menu item.\n", path);
+ }
- return gtk_item_factory_get_widget(factory, path);
+ gtk_widget_set_sensitive(widget, sensitive);
+ g_free(path);
}
-gchar *menu_translate(const gchar *path, gpointer data)
+gchar *cm_menu_item_get_shortcut(GtkUIManager *gui_manager, gchar *menu)
{
- gchar *retval;
+ GtkWidget *widget;
+ gchar *path = g_strdup_printf("/%s/", menu);
+ const gchar *accel = NULL;
+ GtkAccelKey key;
- retval = gettext(path);
+ widget = gtk_ui_manager_get_widget(gui_manager, path);
+ if( !GTK_IS_WIDGET(widget) ) {
+ g_message("Blah, '%s' is not a widget.\n", path);
+ }
+
+ if( !GTK_IS_MENU_ITEM(widget) ) {
+ g_message("Blah, '%s' is not a menu item.\n", path);
+ }
+
+ g_free(path);
+
+ accel = gtk_menu_item_get_accel_path(GTK_MENU_ITEM(widget));
+
+ if (accel && gtk_accel_map_lookup_entry(accel, &key))
+ return gtk_accelerator_get_label (key.accel_key, key.accel_mods);
+ else
+ return g_strdup(_("None"));
- return retval;
}
-static void factory_print_func(gpointer data, gchar *string)
+GtkWidget *cm_menu_item_new_label_from_url(gchar *url)
{
- GString *out_str = data;
-
- g_string_append(out_str, string);
- g_string_append_c(out_str, '\n');
+ gint len = strlen(url);
+ if (len > MAX_MENU_LABEL_LENGTH) {
+ g_message("Refusing a %d bytes string as menu label\n", len);
+ url[64] = '\0', url[63] = url[62] = url[61] = '.', url[60] = ' ';
+ GtkWidget *newlabel = gtk_menu_item_new_with_label(url);
+ gtk_widget_set_tooltip_markup(GTK_WIDGET(newlabel),
+ g_strconcat("<span><b>", _("Warning:"), "</b>",
+ _("This URL was too long for displaying and\n"
+ "has been truncated for safety. This message could be\n"
+ "corrupted, malformed or part of some DoS attempt."),
+ "</span>", NULL));
+ return newlabel;
+ }
+
+ return gtk_menu_item_new_with_label(url);
}
-void menu_set_sensitive(GtkItemFactory *ifactory, const gchar *path,
- gboolean sensitive)
+void cm_toggle_menu_set_active_full(GtkUIManager *gui_manager, gchar *menu, gboolean active)
{
GtkWidget *widget;
+ gchar *path = g_strdup_printf("/%s/", menu);
- g_return_if_fail(ifactory != NULL);
+ widget = gtk_ui_manager_get_widget(gui_manager, path);
+ if( !GTK_IS_WIDGET(widget) ) {
+ g_message("Blah, '%s' is not a widget.\n", path);
+ }
- widget = gtk_item_factory_get_item(ifactory, path);
- g_return_if_fail(widget != NULL);
+ if( !GTK_CHECK_MENU_ITEM(widget) ) {
+ g_message("Blah, '%s' is not a check menu item.\n", path);
+ }
- gtk_widget_set_sensitive(widget, sensitive);
+ gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(widget), active);
+ g_free(path);
}
void menu_set_sensitive_all(GtkMenuShell *menu_shell, gboolean sensitive)
{
+ GList *children = gtk_container_get_children(GTK_CONTAINER(menu_shell));
GList *cur;
- for (cur = menu_shell->children; cur != NULL; cur = cur->next)
+ for (cur = children; cur != NULL; cur = cur->next)
gtk_widget_set_sensitive(GTK_WIDGET(cur->data), sensitive);
-}
-
-void menu_set_active(GtkItemFactory *ifactory, const gchar *path,
- gboolean is_active)
-{
- GtkWidget *widget;
- g_return_if_fail(ifactory != NULL);
-
- widget = gtk_item_factory_get_item(ifactory, path);
- g_return_if_fail(widget != NULL);
-
- if (!GTK_IS_CHECK_MENU_ITEM(widget)) {
- debug_print("%s not check_menu_item\n", path);
- return;
- }
- gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(widget), is_active);
+ g_list_free(children);
}
void menu_button_position(GtkMenu *menu, gint *x, gint *y, gboolean *push_in,
GtkWidget *widget;
gint wheight;
gint wx, wy;
- GtkRequisition mreq;
+ GtkAllocation allocation;
+ GtkRequisition mreq, wreq;
GdkScreen *screen;
GdkRectangle monitor;
gint monitor_num;
- g_return_if_fail(x && y);
- g_return_if_fail(GTK_IS_BUTTON(user_data));
+ cm_return_if_fail(x && y);
+ cm_return_if_fail(GTK_IS_BUTTON(user_data));
widget = GTK_WIDGET(user_data);
- gdk_window_get_origin(widget->window, x, y);
- wheight = widget->requisition.height;
- wx = widget->allocation.x;
- wy = widget->allocation.y;
+ gdk_window_get_origin(gtk_widget_get_window(widget), x, y);
+ gtk_widget_get_requisition(widget, &wreq);
+ wheight = wreq.height;
+ gtk_widget_get_allocation(widget, &allocation);
+ wx = allocation.x;
+ wy = allocation.y;
gtk_widget_size_request(GTK_WIDGET(menu), &mreq);
screen = gtk_widget_get_screen (widget);
*y -= mreq.height;
}
-gint menu_find_option_menu_index(GtkOptionMenu *optmenu, gpointer data,
+gint menu_find_option_menu_index(GtkCMOptionMenu *optmenu, gpointer data,
GCompareFunc func)
{
GtkWidget *menu;
GtkWidget *menuitem;
gpointer menu_data;
+ GList *children;
GList *cur;
- gint n;
+ gint n, found = -1;
- menu = gtk_option_menu_get_menu(optmenu);
+ menu = gtk_cmoption_menu_get_menu(optmenu);
+ children = gtk_container_get_children(GTK_CONTAINER(GTK_MENU_SHELL(menu)));
- for (cur = GTK_MENU_SHELL(menu)->children, n = 0;
+ for (cur = children, n = 0;
cur != NULL; cur = cur->next, n++) {
menuitem = GTK_WIDGET(cur->data);
menu_data = g_object_get_data(G_OBJECT(menuitem),
MENU_VAL_ID);
if (func) {
- if (func(menu_data, data) == 0)
- return n;
- } else if (menu_data == data)
- return n;
- }
-
- return -1;
-}
-
-/* call backs for accelerator changes on selected menu items */
-static void menu_item_add_accel( GtkWidget *widget, guint accel_signal_id, GtkAccelGroup *accel_group,
- guint accel_key, GdkModifierType accel_mods, GtkAccelFlags accel_flags,
- gpointer user_data)
-{
-#warning FIXME_GTK2
-#if 0
- GtkWidget *connected = GTK_WIDGET(user_data);
- if (gtk_signal_n_emissions_by_name(G_OBJECT(widget),"add_accelerator") > 1 ) return;
- gtk_widget_remove_accelerators(connected,"activate",FALSE);
- /* lock _this_ widget */
- gtk_accel_group_lock_entry(accel_group,accel_key,accel_mods);
- /* modify the _other_ widget */
- gtk_widget_add_accelerator(connected, "activate",
- gtk_item_factory_from_widget(connected)->accel_group,
- accel_key, accel_mods,
- GTK_ACCEL_VISIBLE );
- gtk_accel_group_unlock_entry(accel_group,accel_key,accel_mods);
-#endif
-}
-
-static void menu_item_remove_accel(GtkWidget *widget, GtkAccelGroup *accel_group,
- guint accel_key, GdkModifierType accel_mods,
- gpointer user_data)
-{
-#warning FIXME_GTK2
-#if 0
- GtkWidget *wid = GTK_WIDGET(user_data);
-
- if (gtk_signal_n_emissions_by_name(G_OBJECT(widget),
- "remove_accelerator") > 2 )
- return;
- gtk_widget_remove_accelerators(wid,"activate",FALSE);
-#endif
-}
-
-static void connect_accel_change_signals(GtkWidget* widget, GtkWidget *wid2)
-{
-#warning FIXME_GTK2
-#if 0
- g_signal_connect_after(G_OBJECT(widget), "add_accelerator",
- G_CALLBACK(menu_item_add_accel), wid2);
- g_signal_connect_after(G_OBJECT(widget), "remove_accelerator",
- G_CALLBACK(menu_item_remove_accel), wid2);
-#endif
-}
-
-void menu_connect_identical_items(void)
-{
- gint n;
- GtkWidget *item1;
- GtkWidget *item2;
-
- static const struct {
- const gchar *path1;
- const gchar *path2;
- } pairs[] = {
- {"<Main>/Message/Reply", "<SummaryView>/Reply"},
- {"<Main>/Message/Reply to/all", "<SummaryView>/Reply to/all"},
- {"<Main>/Message/Reply to/sender", "<SummaryView>/Reply to/sender"},
- {"<Main>/Message/Reply to/mailing list", "<SummaryView>/Reply to/mailing list"},
- {"<Main>/Message/Follow-up and reply to", "<SummaryView>/Follow-up and reply to"},
- {"<Main>/Message/Forward", "<SummaryView>/Forward"},
- {"<Main>/Message/Redirect", "<SummaryView>/Redirect"},
- {"<Main>/Message/Re-edit", "<SummaryView>/Re-edit"},
- {"<Main>/Message/Move...", "<SummaryView>/Move..."},
- {"<Main>/Message/Copy...", "<SummaryView>/Copy..."},
- {"<Main>/Message/Delete", "<SummaryView>/Delete"},
- {"<Main>/Message/Cancel a news message", "<SummaryView>/Cancel a news message"},
- {"<Main>/Message/Mark/Mark", "<SummaryView>/Mark/Mark"},
- {"<Main>/Message/Mark/Unmark", "<SummaryView>/Mark/Unmark"},
- {"<Main>/Message/Mark/Mark as unread", "<SummaryView>/Mark/Mark as unread"},
- {"<Main>/Message/Mark/Mark as read", "<SummaryView>/Mark/Mark as read"},
- {"<Main>/Message/Mark/Mark all read", "<SummaryView>/Mark/Mark all read"},
- {"<Main>/Tools/Add sender to address book", "<SummaryView>/Add sender to address book"},
- {"<Main>/Tools/Create filter rule/Automatically",
- "<SummaryView>/Create filter rule/Automatically"},
- {"<Main>/Tools/Create filter rule/by From", "<SummaryView>/Create filter rule/by From"},
- {"<Main>/Tools/Create filter rule/by To", "<SummaryView>/Create filter rule/by To"},
- {"<Main>/Tools/Create filter rule/by Subject", "<SummaryView>/Create filter rule/by Subject"},
- {"<Main>/Tools/Create processing rule/Automatically",
- "<SummaryView>/Create processing rule/Automatically"},
- {"<Main>/Tools/Create processing rule/by From", "<SummaryView>/Create processing rule/by From"},
- {"<Main>/Tools/Create processing rule/by To", "<SummaryView>/Create processing rule/by To"},
- {"<Main>/Tools/Create processing rule/by Subject",
- "<SummaryView>/Create processing rule/by Subject"},
- {"<Main>/View/Open in new window", "<SummaryView>/View/Open in new window"},
- {"<Main>/View/Message source", "<SummaryView>/View/Source"},
- {"<Main>/View/Show all headers", "<SummaryView>/View/All header"},
- };
-
- const gint numpairs = sizeof pairs / sizeof pairs[0];
- for (n = 0; n < numpairs; n++) {
- /* get widgets from the paths */
-
- item1 = gtk_item_factory_get_widget
- (gtk_item_factory_from_path(pairs[n].path1),pairs[n].path1);
- item2 = gtk_item_factory_get_widget
- (gtk_item_factory_from_path(pairs[n].path2),pairs[n].path2);
-
- if (item1 && item2) {
- /* connect widgets both ways around */
- connect_accel_change_signals(item2,item1);
- connect_accel_change_signals(item1,item2);
- } else {
- if (!item1) debug_print(" ** Menu item not found: %s\n",pairs[n].path1);
- if (!item2) debug_print(" ** Menu item not found: %s\n",pairs[n].path2);
- }
- }
-}
-
-void menu_select_by_data(GtkMenu *menu, gpointer data)
-{
- GList *children, *cur;
- GtkWidget *select_item = NULL;
-
- g_return_if_fail(menu != NULL);
-
- children = gtk_container_get_children(GTK_CONTAINER(menu));
-
- for (cur = children; cur != NULL; cur = g_list_next(cur)) {
- GObject *child = G_OBJECT(cur->data);
-
- if (g_object_get_data(child, MENU_VAL_ID) == data) {
- select_item = GTK_WIDGET(child);
+ if (func(menu_data, data) == 0) {
+ found = n;
+ break;
+ }
+ } else if (menu_data == data) {
+ found = n;
+ break;
}
}
- if (select_item != NULL) {
- gtk_menu_shell_select_item(GTK_MENU_SHELL(menu), select_item);
- gtk_menu_shell_activate_item(GTK_MENU_SHELL(menu), select_item, FALSE);
- }
g_list_free(children);
+
+ return found;
}