1 /*
2  * Sylpheed -- a GTK+ based, lightweight, and fast e-mail client
3  * Copyright (C) 1999-2013 Hiroyuki Yamamoto and the Claws Mail team
4  *
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 3 of the License, or
8  * (at your option) any later version.
9  *
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.
14  *
15  * You should have received a copy of the GNU General Public License
16  * along with this program. If not, see <http://www.gnu.org/licenses/>.
17  *
18  */
19 
20 #ifdef HAVE_CONFIG_H
21 #  include "config.h"
22 #include "claws-features.h"
23 #endif
24 
25 #include <glib.h>
26 #include <glib/gi18n.h>
27 #include <gtk/gtk.h>
28 
29 #include "gtkcmoptionmenu.h"
30 #include "menu.h"
31 #include "utils.h"
32 #include "gtkutils.h"
33 #include "defs.h"
34 
cm_menu_create_action_group(const gchar * name,GtkActionEntry * entries,gint num_entries,gpointer data)35 GtkActionGroup *cm_menu_create_action_group(const gchar *name, GtkActionEntry *entries,
36 					    gint num_entries, gpointer data)
37 {
38 	GtkActionGroup *group = gtk_action_group_new(name);
39 	gtk_action_group_set_translate_func(group, menu_translate, NULL, NULL);
40 	gtk_action_group_add_actions(group, entries, num_entries, data);
41 	gtk_ui_manager_insert_action_group(gtkut_ui_manager(), group, 0);
42 	return group;
43 }
44 
cm_menu_create_action_group_full(GtkUIManager * manager,const gchar * name,GtkActionEntry * entries,gint num_entries,gpointer data)45 GtkActionGroup *cm_menu_create_action_group_full(GtkUIManager *manager, const gchar *name, GtkActionEntry *entries,
46 					    gint num_entries, gpointer data)
47 {
48 	GtkActionGroup *group = gtk_action_group_new(name);
49 	gtk_action_group_set_translate_func(group, menu_translate, NULL, NULL);
50 	gtk_action_group_add_actions(group, entries, num_entries, data);
51 	gtk_ui_manager_insert_action_group(manager, group, 0);
52 	return group;
53 }
54 
menu_translate(const gchar * path,gpointer data)55 gchar *menu_translate(const gchar *path, gpointer data)
56 {
57 	gchar *retval;
58 
59 	retval = gettext(path);
60 
61 	return retval;
62 }
63 
cm_menu_set_sensitive(gchar * menu,gboolean sensitive)64 void cm_menu_set_sensitive(gchar *menu, gboolean sensitive)
65 {
66 	GtkUIManager *gui_manager = gtkut_ui_manager();
67 	gchar *path = g_strdup_printf("Menus/%s", menu);
68 
69 	cm_menu_set_sensitive_full(gui_manager, path, sensitive);
70 	g_free(path);
71 }
72 
cm_toggle_menu_set_active(gchar * menu,gboolean active)73 void cm_toggle_menu_set_active(gchar *menu, gboolean active)
74 {
75 	GtkUIManager *gui_manager = gtkut_ui_manager();
76 	gchar *path = g_strdup_printf("Menus/%s", menu);
77 
78 	cm_toggle_menu_set_active_full(gui_manager, path, active);
79 	g_free(path);
80 }
81 
cm_menu_set_sensitive_full(GtkUIManager * gui_manager,const gchar * menu,gboolean sensitive)82 void cm_menu_set_sensitive_full(GtkUIManager *gui_manager, const gchar *menu, gboolean sensitive)
83 {
84 	GtkWidget *widget;
85 	gchar *path = g_strdup_printf("/%s/", menu);
86 
87 	widget = gtk_ui_manager_get_widget(gui_manager, path);
88 	if( !GTK_IS_WIDGET(widget) ) {
89 		g_message("Blah, '%s' is not a widget.\n", path);
90 	}
91 
92 	if( !GTK_IS_MENU_ITEM(widget) ) {
93 		g_message("Blah, '%s' is not a menu item.\n", path);
94 	}
95 
96 	gtk_widget_set_sensitive(widget, sensitive);
97 
98 	if (strcmp(menu, "Menus/SummaryViewPopup/Reedit") == 0)
99 		(sensitive)? gtk_widget_show(widget) : gtk_widget_hide(widget);
100 
101 	g_free(path);
102 }
103 
cm_menu_item_get_shortcut(GtkUIManager * gui_manager,gchar * menu)104 gchar *cm_menu_item_get_shortcut(GtkUIManager *gui_manager, gchar *menu)
105 {
106 	GtkWidget *widget;
107 	gchar *path = g_strdup_printf("/%s/", menu);
108 	const gchar *accel = NULL;
109 	GtkAccelKey key;
110 
111 	widget = gtk_ui_manager_get_widget(gui_manager, path);
112 	if( !GTK_IS_WIDGET(widget) ) {
113 		g_message("Blah, '%s' is not a widget.\n", path);
114 	}
115 
116 	if( !GTK_IS_MENU_ITEM(widget) ) {
117 		g_message("Blah, '%s' is not a menu item.\n", path);
118 	}
119 
120 	g_free(path);
121 
122 	accel = gtk_menu_item_get_accel_path(GTK_MENU_ITEM(widget));
123 
124 	if (accel && gtk_accel_map_lookup_entry(accel, &key))
125 		return gtk_accelerator_get_label (key.accel_key, key.accel_mods);
126 	else
127 		return g_strdup(_("None"));
128 
129 }
130 
cm_menu_item_new_label_from_url(gchar * url)131 GtkWidget *cm_menu_item_new_label_from_url(gchar *url)
132 {
133 	gint len = strlen(url);
134 	if (len > MAX_MENU_LABEL_LENGTH) {
135 		g_message("Refusing a %d bytes string as menu label\n", len);
136 		url[64] = '\0', url[63] = url[62] = url[61] = '.', url[60] = ' ';
137 		GtkWidget *newlabel = gtk_menu_item_new_with_label(url);
138 		gtk_widget_set_tooltip_markup(GTK_WIDGET(newlabel),
139 			g_strconcat("<span><b>", _("Warning:"), "</b>",
140 			_("This URL was too long for displaying and\n"
141 			"has been truncated for safety. This message could be\n"
142 			"corrupted, malformed or part of some DoS attempt."),
143 			"</span>", NULL));
144 		return newlabel;
145 	}
146 
147 	return gtk_menu_item_new_with_label(url);
148 }
149 
cm_toggle_menu_set_active_full(GtkUIManager * gui_manager,gchar * menu,gboolean active)150 void cm_toggle_menu_set_active_full(GtkUIManager *gui_manager, gchar *menu, gboolean active)
151 {
152 	GtkWidget *widget;
153 	gchar *path = g_strdup_printf("/%s/", menu);
154 
155 	widget = gtk_ui_manager_get_widget(gui_manager, path);
156 	if( !GTK_IS_WIDGET(widget) ) {
157 		g_message("Blah, '%s' is not a widget.\n", path);
158 	}
159 
160 	if( !GTK_CHECK_MENU_ITEM(widget) ) {
161 		g_message("Blah, '%s' is not a check menu item.\n", path);
162 	}
163 
164 	gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(widget), active);
165 	g_free(path);
166 }
167 
menu_set_sensitive_all(GtkMenuShell * menu_shell,gboolean sensitive)168 void menu_set_sensitive_all(GtkMenuShell *menu_shell, gboolean sensitive)
169 {
170 	GList *children = gtk_container_get_children(GTK_CONTAINER(menu_shell));
171 	GList *cur;
172 
173 	for (cur = children; cur != NULL; cur = cur->next)
174 		gtk_widget_set_sensitive(GTK_WIDGET(cur->data), sensitive);
175 
176 	g_list_free(children);
177 }
178 
menu_button_position(GtkMenu * menu,gint * x,gint * y,gboolean * push_in,gpointer user_data)179 void menu_button_position(GtkMenu *menu, gint *x, gint *y, gboolean *push_in,
180 			  gpointer user_data)
181 {
182         GtkWidget *widget;
183         gint wheight;
184         gint wx, wy;
185 	GtkAllocation allocation;
186 	GtkRequisition mreq, wreq;
187 	GdkScreen *screen;
188 	GdkRectangle monitor;
189 	gint monitor_num;
190 
191 	cm_return_if_fail(x && y);
192  	cm_return_if_fail(GTK_IS_BUTTON(user_data));
193 
194 	widget = GTK_WIDGET(user_data);
195 
196         gdk_window_get_origin(gtk_widget_get_window(widget), x, y);
197         gtk_widget_get_requisition(widget, &wreq);
198         wheight = wreq.height;
199         gtk_widget_get_allocation(widget, &allocation);
200         wx = allocation.x;
201         wy = allocation.y;
202 
203 	gtk_widget_size_request(GTK_WIDGET(menu), &mreq);
204 	screen = gtk_widget_get_screen (widget);
205 	monitor_num = gdk_screen_get_monitor_at_point (screen, *x, *y);
206 	gdk_screen_get_monitor_geometry (screen, monitor_num,
207 					 &monitor);
208 
209         *x = *x + wx;
210         *y = *y + wy + wheight;
211 
212 	if (*y + mreq.height >= monitor.height)
213 		*y -= mreq.height;
214 }
215 
menu_find_option_menu_index(GtkCMOptionMenu * optmenu,gpointer data,GCompareFunc func)216 gint menu_find_option_menu_index(GtkCMOptionMenu *optmenu, gpointer data,
217 				 GCompareFunc func)
218 {
219 	GtkWidget *menu;
220 	GtkWidget *menuitem;
221 	gpointer menu_data;
222 	GList *children;
223 	GList *cur;
224 	gint n, found = -1;
225 
226 	menu = gtk_cmoption_menu_get_menu(optmenu);
227 	children = gtk_container_get_children(GTK_CONTAINER(GTK_MENU_SHELL(menu)));
228 
229 	for (cur = children, n = 0;
230 	     cur != NULL; cur = cur->next, n++) {
231 		menuitem = GTK_WIDGET(cur->data);
232 		menu_data = g_object_get_data(G_OBJECT(menuitem),
233 					      MENU_VAL_ID);
234 		if (func) {
235 			if (func(menu_data, data) == 0) {
236 				found = n;
237 				break;
238 			}
239 		} else if (menu_data == data) {
240 			found = n;
241 			break;
242 		}
243 	}
244 
245 	g_list_free(children);
246 
247 	return found;
248 }
249