2 * ROX-Filer, filer for the ROX desktop project
3 * Copyright (C) 2006, Thomas Leonard and others (see changelog for details).
5 * This program is free software; you can redistribute it and/or modify it
6 * under the terms of the GNU General Public License as published by the Free
7 * Software Foundation; either version 2 of the License, or (at your option)
10 * This program is distributed in the hope that it will be useful, but WITHOUT
11 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
12 * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
15 * You should have received a copy of the GNU General Public License along with
16 * this program; if not, write to the Free Software Foundation, Inc., 59 Temple
17 * Place, Suite 330, Boston, MA 02111-1307 USA
20 /* appmenu.c - handles application-specific menus read from XMLwrapper.xml */
22 /* XXX: This handles all File menu extensions. Needs renaming! */
32 #include <libxml/parser.h>
38 #include "gui_support.h"
52 /* Static prototypes */
53 static void apprun_menu(GtkWidget
*item
, gpointer data
);
54 static GtkWidget
*create_menu_item(xmlNode
*node
);
55 static void show_app_help(GtkWidget
*item
, gpointer data
);
56 static void build_app_menu(const char *app_dir
, DirItem
*app_item
);
57 static void mnt_eject(GtkWidget
*item
, gpointer data
);
59 /* There can only be one menu open at a time... we store: */
60 static GtkWidget
*current_menu
= NULL
; /* The GtkMenu */
61 static guchar
*current_app_path
= NULL
; /* The path of the application */
62 static GList
*current_items
= NULL
; /* The GtkMenuItems we added directly
63 * to it --- not submenu items.
65 /****************************************************************
66 * EXTERNAL INTERFACE *
67 ****************************************************************/
69 /* Removes all appmenu menu items */
70 void appmenu_remove(void)
77 for (next
= current_items
; next
; next
= next
->next
)
78 gtk_widget_destroy((GtkWidget
*) next
->data
);
80 null_g_free(¤t_app_path
);
83 g_list_free(current_items
);
87 /* Add AppMenu entries to 'menu', if appropriate.
88 * This function modifies the menu stored in "menu".
89 * 'app_dir' is the pathname of the application directory, and 'item'
90 * is the corresponding DirItem.
91 * Returns number of entries added.
92 * Call appmenu_remove() to undo the effect.
94 int appmenu_add(const gchar
*app_dir
, DirItem
*app_item
, GtkWidget
*menu
)
100 g_return_val_if_fail(menu
!= NULL
, 0);
102 /* Should have called appmenu_remove() already... */
103 g_return_val_if_fail(current_menu
== NULL
, 0);
104 g_return_val_if_fail(current_items
== NULL
, 0);
106 build_app_menu(app_dir
, app_item
);
108 if (app_item
->flags
& ITEM_FLAG_MOUNT_POINT
)
111 item
= gtk_menu_item_new_with_label(_("Eject"));
112 gtk_widget_show(item
);
113 current_items
= g_list_prepend(current_items
, item
);
114 g_signal_connect(item
, "activate", G_CALLBACK(mnt_eject
), NULL
);
119 sep
= gtk_menu_item_new();
120 current_items
= g_list_prepend(current_items
, sep
);
121 gtk_widget_show(sep
);
124 for (next
= current_items
; next
; next
= next
->next
)
126 gtk_menu_shell_prepend(GTK_MENU_SHELL(menu
),
127 GTK_WIDGET(next
->data
));
132 current_app_path
= g_strdup(app_dir
);
138 /****************************************************************
139 * INTERNAL FUNCTIONS *
140 ****************************************************************/
142 /* Create a new menu and return it */
143 static GtkWidget
*appmenu_add_submenu(xmlNode
*subm_node
)
148 /* Create the new submenu */
149 sub_menu
= gtk_menu_new();
151 /* Add the menu entries */
152 for (node
= subm_node
->xmlChildrenNode
; node
; node
= node
->next
)
156 item
= create_menu_item(node
);
158 gtk_menu_shell_append(GTK_MENU_SHELL(sub_menu
), item
);
164 /* Create and return a menu item */
165 static GtkWidget
*create_menu_item(xmlNode
*node
)
169 guchar
*label
, *option
= NULL
;
172 if (node
->type
!= XML_ELEMENT_NODE
)
175 if (strcmp(node
->name
, "Item") == 0)
178 option
= xmlGetProp(node
, "option");
180 else if (strcmp(node
->name
, "AppMenu") == 0)
185 /* Create the item */
186 label_node
= get_subnode(node
, NULL
, "Label");
189 label
= xmlNodeListGetString(label_node
->doc
,
190 label_node
->xmlChildrenNode
, 1);
194 label
= xmlGetProp(node
, "label");
196 label
= g_strdup(_("<missing label>"));
198 item
= gtk_menu_item_new_with_label(label
);
200 gtk_widget_set_accel_path(item
, NULL
, NULL
); /* XXX */
206 /* Add submenu items */
208 gtk_menu_item_set_submenu(GTK_MENU_ITEM(item
),
209 appmenu_add_submenu(node
));
213 /* Set up callback */
217 g_object_set_data_full(G_OBJECT(item
), "option",
223 g_signal_connect(item
, "activate", G_CALLBACK(apprun_menu
),
227 gtk_widget_show(item
);
232 /* Send to current_app_path (though not actually an app) */
233 static void send_to(GtkWidget
*item
, const char *app
)
237 g_return_if_fail(current_app_path
!= NULL
);
239 file_list
= g_list_prepend(NULL
, current_app_path
);
240 run_with_files(app
, file_list
);
241 g_list_free(file_list
);
244 /* Function called to execute an AppMenu item */
245 static void apprun_menu(GtkWidget
*item
, gpointer data
)
250 g_return_if_fail(current_app_path
!= NULL
);
252 option
= g_object_get_data(G_OBJECT(item
), "option");
254 argv
[0] = g_strconcat(current_app_path
, "/AppRun", NULL
);
255 argv
[1] = option
; /* (may be NULL) */
258 rox_spawn(NULL
, (const gchar
**) argv
);
263 static void show_app_help(GtkWidget
*item
, gpointer data
)
265 g_return_if_fail(current_app_path
!= NULL
);
267 show_help_files(current_app_path
);
270 static void mnt_eject(GtkWidget
*item
, gpointer data
)
274 g_return_if_fail(current_app_path
!= NULL
);
275 dirs
= g_list_prepend(NULL
, current_app_path
);
280 static void customise_type(GtkWidget
*item
, MIME_type
*type
)
285 leaf
= g_strconcat(".", type
->media_type
, "_", type
->subtype
, NULL
);
286 path
= choices_find_xdg_path_save(leaf
, "SendTo", SITE
, TRUE
);
290 filer_opendir(path
, NULL
, NULL
);
293 info_message(_("Symlink any programs you want into this directory. "
294 "They will appear in the menu for all items of this "
295 "type (%s/%s)."), type
->media_type
, type
->subtype
);
298 static void build_menu_for_type(MIME_type
*type
)
307 leaf
= g_strconcat(".", type
->media_type
, "_", type
->subtype
, NULL
);
308 path
= choices_find_xdg_path_load(leaf
, "SendTo", SITE
);
313 names
= list_dir(path
);
315 ditem
= diritem_new("");
317 for (i
= 0; i
< names
->len
; i
++)
319 char *leaf
= names
->pdata
[i
];
322 full_path
= g_build_filename(path
, leaf
, NULL
);
323 diritem_restat(full_path
, ditem
, NULL
);
325 item
= make_send_to_item(ditem
, leaf
, MIS_SMALL
);
326 current_items
= g_list_prepend(current_items
, item
);
327 gtk_widget_show(item
);
328 g_signal_connect_data(item
, "activate", G_CALLBACK(send_to
),
329 full_path
, (GClosureNotify
) g_free
, 0);
332 g_ptr_array_free(names
, TRUE
);
337 item
= gtk_menu_item_new_with_label(_("Customise Menu..."));
338 current_items
= g_list_prepend(current_items
, item
);
339 g_signal_connect(item
, "activate", G_CALLBACK(customise_type
), type
);
340 gtk_widget_show(item
);
345 static inline gboolean
is_dir(const char *dir
)
348 return stat(dir
, &info
) == 0 && S_ISDIR(info
.st_mode
);
351 /* Adds to current_items */
352 static void build_app_menu(const char *app_dir
, DirItem
*app_item
)
354 XMLwrapper
*ai
= NULL
;
359 ai
= appinfo_get(app_dir
, app_item
);
362 node
= xml_get_section(ai
, NULL
, "AppMenu");
364 node
= node
->xmlChildrenNode
;
368 if (app_item
->flags
& ITEM_FLAG_APPDIR
)
372 /* Not an application AND no AppInfo */
373 build_menu_for_type(app_item
->mime_type
);
378 /* Add the menu entries */
379 for (; node
; node
= node
->next
)
381 item
= create_menu_item(node
);
384 current_items
= g_list_prepend(current_items
, item
);
387 help_dir
= g_build_filename(app_dir
, "Help", NULL
);
388 if (is_dir(help_dir
))
390 item
= gtk_image_menu_item_new_from_stock(GTK_STOCK_HELP
, NULL
);
391 gtk_widget_show(item
);
392 current_items
= g_list_prepend(current_items
, item
);
393 g_signal_connect(item
, "activate", G_CALLBACK(show_app_help
), NULL
);
394 gtk_label_set_text(GTK_LABEL(GTK_BIN(item
)->child
), _("Help"));