2 * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3 * Copyright (C) 1999-2004 Hiroyuki Yamamoto
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License as published by
7 * the Free Software Foundation; either version 2 of the License, or
8 * (at your option) any later version.
10 * This program is distributed in the hope that it will be useful,
11 * but WITHOUT ANY WARRANTY; without even the implied warranty of
12 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 * GNU General Public License for more details.
15 * You should have received a copy of the GNU General Public License
16 * along with this program; if not, write to the Free Software
17 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
26 #include <gtk/gtkwidget.h>
27 #include <gtk/gtkmenu.h>
28 #include <gtk/gtkmenubar.h>
29 #include <gtk/gtkcheckmenuitem.h>
30 #include <gtk/gtkitemfactory.h>
31 #include <gtk/gtkbutton.h>
32 #include <gtk/gtkwindow.h>
38 static void menu_item_add_accel( GtkWidget *widget, guint accel_signal_id, GtkAccelGroup *accel_group,
39 guint accel_key, GdkModifierType accel_mods, GtkAccelFlags accel_flags,
42 static void menu_item_remove_accel(GtkWidget *widget, GtkAccelGroup *accel_group,
43 guint accel_key, GdkModifierType accel_mods,
46 static void connect_accel_change_signals(GtkWidget* widget, GtkWidget *wid2) ;
49 GtkWidget *menubar_create(GtkWidget *window, GtkItemFactoryEntry *entries,
50 guint n_entries, const gchar *path, gpointer data)
52 GtkItemFactory *factory;
54 factory = gtk_item_factory_new(GTK_TYPE_MENU_BAR, path, NULL);
55 gtk_item_factory_set_translate_func(factory, menu_translate,
57 gtk_item_factory_create_items(factory, n_entries, entries, data);
58 gtk_window_add_accel_group (GTK_WINDOW (window), factory->accel_group);
60 return gtk_item_factory_get_widget(factory, path);
63 GtkWidget *menu_create_items(GtkItemFactoryEntry *entries,
64 guint n_entries, const gchar *path,
65 GtkItemFactory **factory, gpointer data)
67 *factory = gtk_item_factory_new(GTK_TYPE_MENU, path, NULL);
68 gtk_item_factory_set_translate_func(*factory, menu_translate,
70 gtk_item_factory_create_items(*factory, n_entries, entries, data);
72 return gtk_item_factory_get_widget(*factory, path);
75 GtkWidget *popupmenu_create(GtkWidget *window, GtkItemFactoryEntry *entries,
76 guint n_entries, const gchar *path, gpointer data)
78 GtkItemFactory *factory;
79 GtkAccelGroup *accel_group;
81 accel_group = gtk_accel_group_new();
82 factory = gtk_item_factory_new(GTK_TYPE_MENU, path, accel_group);
83 gtk_item_factory_set_translate_func(factory, menu_translate,
85 gtk_item_factory_create_items(factory, n_entries, entries, data);
86 gtk_window_add_accel_group(GTK_WINDOW (window), accel_group);
88 return gtk_item_factory_get_widget(factory, path);
91 gchar *menu_translate(const gchar *path, gpointer data)
95 retval = gettext(path);
100 static void factory_print_func(gpointer data, gchar *string)
102 GString *out_str = data;
104 g_string_append(out_str, string);
105 g_string_append_c(out_str, '\n');
108 void menu_set_sensitive(GtkItemFactory *ifactory, const gchar *path,
113 g_return_if_fail(ifactory != NULL);
115 widget = gtk_item_factory_get_item(ifactory, path);
117 debug_print("unknown menu entry %s\n", path);
120 gtk_widget_set_sensitive(widget, sensitive);
123 void menu_set_sensitive_all(GtkMenuShell *menu_shell, gboolean sensitive)
127 for (cur = menu_shell->children; cur != NULL; cur = cur->next)
128 gtk_widget_set_sensitive(GTK_WIDGET(cur->data), sensitive);
131 void menu_set_active(GtkItemFactory *ifactory, const gchar *path,
136 g_return_if_fail(ifactory != NULL);
138 widget = gtk_item_factory_get_item(ifactory, path);
139 if (!GTK_IS_CHECK_MENU_ITEM(widget)) {
140 debug_print("%s not check_menu_item\n", path);
143 gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(widget), is_active);
146 void menu_button_position(GtkMenu *menu, gint *x, gint *y, gboolean *push_in,
154 GdkRectangle monitor;
157 g_return_if_fail(x && y);
158 g_return_if_fail(GTK_IS_BUTTON(user_data));
160 widget = GTK_WIDGET(user_data);
162 gdk_window_get_origin(widget->window, x, y);
163 wheight = widget->requisition.height;
164 wx = widget->allocation.x;
165 wy = widget->allocation.y;
167 gtk_widget_size_request(GTK_WIDGET(menu), &mreq);
168 screen = gtk_widget_get_screen (widget);
169 monitor_num = gdk_screen_get_monitor_at_point (screen, *x, *y);
170 gdk_screen_get_monitor_geometry (screen, monitor_num,
174 *y = *y + wy + wheight;
176 if (*y + mreq.height >= monitor.height)
180 gint menu_find_option_menu_index(GtkOptionMenu *optmenu, gpointer data,
189 menu = gtk_option_menu_get_menu(optmenu);
191 for (cur = GTK_MENU_SHELL(menu)->children, n = 0;
192 cur != NULL; cur = cur->next, n++) {
193 menuitem = GTK_WIDGET(cur->data);
194 menu_data = g_object_get_data(G_OBJECT(menuitem),
197 if (func(menu_data, data) == 0)
199 } else if (menu_data == data)
206 /* call backs for accelerator changes on selected menu items */
207 static void menu_item_add_accel( GtkWidget *widget, guint accel_signal_id, GtkAccelGroup *accel_group,
208 guint accel_key, GdkModifierType accel_mods, GtkAccelFlags accel_flags,
213 GtkWidget *connected = GTK_WIDGET(user_data);
214 if (gtk_signal_n_emissions_by_name(G_OBJECT(widget),"add_accelerator") > 1 ) return;
215 gtk_widget_remove_accelerators(connected,"activate",FALSE);
216 /* lock _this_ widget */
217 gtk_accel_group_lock_entry(accel_group,accel_key,accel_mods);
218 /* modify the _other_ widget */
219 gtk_widget_add_accelerator(connected, "activate",
220 gtk_item_factory_from_widget(connected)->accel_group,
221 accel_key, accel_mods,
223 gtk_accel_group_unlock_entry(accel_group,accel_key,accel_mods);
227 static void menu_item_remove_accel(GtkWidget *widget, GtkAccelGroup *accel_group,
228 guint accel_key, GdkModifierType accel_mods,
233 GtkWidget *wid = GTK_WIDGET(user_data);
235 if (gtk_signal_n_emissions_by_name(G_OBJECT(widget),
236 "remove_accelerator") > 2 )
238 gtk_widget_remove_accelerators(wid,"activate",FALSE);
242 static void connect_accel_change_signals(GtkWidget* widget, GtkWidget *wid2)
246 g_signal_connect_after(G_OBJECT(widget), "add_accelerator",
247 G_CALLBACK(menu_item_add_accel), wid2);
248 g_signal_connect_after(G_OBJECT(widget), "remove_accelerator",
249 G_CALLBACK(menu_item_remove_accel), wid2);
253 void menu_connect_identical_items(void)
259 static const struct {
263 {"<Main>/Message/Reply", "<SummaryView>/Reply"},
264 {"<Main>/Message/Reply to/all", "<SummaryView>/Reply to/all"},
265 {"<Main>/Message/Reply to/sender", "<SummaryView>/Reply to/sender"},
266 {"<Main>/Message/Reply to/mailing list", "<SummaryView>/Reply to/mailing list"},
267 {"<Main>/Message/Follow-up and reply to", "<SummaryView>/Follow-up and reply to"},
268 {"<Main>/Message/Forward", "<SummaryView>/Forward"},
269 {"<Main>/Message/Redirect", "<SummaryView>/Redirect"},
270 {"<Main>/Message/Re-edit", "<SummaryView>/Re-edit"},
271 {"<Main>/Message/Move...", "<SummaryView>/Move..."},
272 {"<Main>/Message/Copy...", "<SummaryView>/Copy..."},
273 {"<Main>/Message/Delete", "<SummaryView>/Delete"},
274 {"<Main>/Message/Cancel a news message", "<SummaryView>/Cancel a news message"},
275 {"<Main>/Message/Mark/Mark", "<SummaryView>/Mark/Mark"},
276 {"<Main>/Message/Mark/Unmark", "<SummaryView>/Mark/Unmark"},
277 {"<Main>/Message/Mark/Mark as unread", "<SummaryView>/Mark/Mark as unread"},
278 {"<Main>/Message/Mark/Mark as read", "<SummaryView>/Mark/Mark as read"},
279 {"<Main>/Message/Mark/Mark all read", "<SummaryView>/Mark/Mark all read"},
280 {"<Main>/Tools/Add sender to address book", "<SummaryView>/Add sender to address book"},
281 {"<Main>/Tools/Create filter rule/Automatically", "<SummaryView>/Create filter rule/Automatically"},
282 {"<Main>/Tools/Create filter rule/by From", "<SummaryView>/Create filter rule/by From"},
283 {"<Main>/Tools/Create filter rule/by To", "<SummaryView>/Create filter rule/by To"},
284 {"<Main>/Tools/Create filter rule/by Subject", "<SummaryView>/Create filter rule/by Subject"},
285 {"<Main>/View/Open in new window", "<SummaryView>/View/Open in new window"},
286 {"<Main>/View/Message source", "<SummaryView>/View/Source"},
287 {"<Main>/View/Show all headers", "<SummaryView>/View/All header"},
290 const gint numpairs = sizeof pairs / sizeof pairs[0];
291 for (n = 0; n < numpairs; n++) {
292 /* get widgets from the paths */
294 item1 = gtk_item_factory_get_widget
295 (gtk_item_factory_from_path(pairs[n].path1),pairs[n].path1);
296 item2 = gtk_item_factory_get_widget
297 (gtk_item_factory_from_path(pairs[n].path2),pairs[n].path2);
299 if (item1 && item2) {
300 /* connect widgets both ways around */
301 connect_accel_change_signals(item2,item1);
302 connect_accel_change_signals(item1,item2);
304 if (!item1) debug_print(" ** Menu item not found: %s\n",pairs[n].path1);
305 if (!item2) debug_print(" ** Menu item not found: %s\n",pairs[n].path2);
310 void menu_select_by_data(GtkMenu *menu, gpointer data)
312 GList *children, *cur;
313 GtkWidget *select_item = NULL;
315 g_return_if_fail(menu != NULL);
317 children = gtk_container_children(GTK_CONTAINER(menu));
319 for (cur = children; cur != NULL; cur = g_list_next(cur)) {
320 GtkObject *child = G_OBJECT(cur->data);
322 if (gtk_object_get_user_data(child) == data) {
323 select_item = GTK_WIDGET(child);
326 if (select_item != NULL) {
327 gtk_menu_shell_select_item(GTK_MENU_SHELL(menu), select_item);
328 gtk_menu_shell_activate_item(GTK_MENU_SHELL(menu), select_item, FALSE);
331 g_list_free(children);