Desktop
/* $Id: callbacks.c,v 1.21 2006/11/15 23:17:42 khorben Exp $ */
/* Copyright (c) 2006 The DeforaOS Project */
#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <errno.h>
#include "callbacks.h"
#include "browser.h"
#include "../config.h"
/* constants */
static char const * _authors[] =
{
"Pierre 'khorben' Pronchery",
NULL
};
/* FIXME */
static char const _license[] = "GPLv2";
/* callbacks */
/* window */
gboolean on_closex(GtkWidget * widget, GdkEvent * event, gpointer data)
{
gtk_widget_hide(widget);
gtk_main_quit();
return FALSE;
}
/* file menu */
void on_file_new_window(GtkMenuItem * menuitem, gpointer data)
{
Browser * browser = data;
pid_t pid;
if((pid = fork()) == -1)
{
browser_error(browser, strerror(errno), 0);
return;
}
if(pid != 0)
return;
execlp("browser", "browser", browser->current->data, NULL);
fprintf(stderr, "%s%s\n", "browser: browser: ", strerror(errno));
exit(2);
}
void on_file_close(GtkMenuItem * menuitem, gpointer data)
{
gtk_main_quit();
}
/* edit menu */
static GList * _copy_selection(Browser * browser);
void on_edit_copy(GtkMenuItem * menuitem, gpointer data)
/* FIXME */
{
Browser * browser = data;
GtkTreeIter iter;
GList * sel;
GList * p;
gchar * q;
if((sel = _copy_selection(browser)) == NULL)
return;
for(p = sel; p != NULL; p = p->next)
{
if(!gtk_tree_model_get_iter(GTK_TREE_MODEL(browser->store),
&iter, p->data))
continue;
gtk_tree_model_get(GTK_TREE_MODEL(browser->store), &iter,
BR_COL_PATH, &q, -1);
printf("%s\n", q);
g_free(q);
}
g_list_foreach(sel, (GFunc)gtk_tree_path_free, NULL);
g_list_free(sel);
}
static GList * _copy_selection(Browser * browser)
{
#if GTK_CHECK_VERSION(2, 6, 0)
if(browser->iconview != NULL)
return gtk_icon_view_get_selected_items(GTK_ICON_VIEW(
browser->iconview));
else
#endif
{
GtkTreeSelection * treesel;
if((treesel = gtk_tree_view_get_selection(GTK_TREE_VIEW(
browser->detailview)))
== NULL)
return NULL;
return gtk_tree_selection_get_selected_rows(treesel, NULL);
}
}
void on_edit_cut(GtkMenuItem * menuitem, gpointer data)
/* FIXME */
{
Browser * browser = data;
GtkTreeIter iter;
GList * sel;
GList * p;
gchar * q;
if((sel = _copy_selection(browser)) == NULL)
return;
for(p = sel; p != NULL; p = p->next)
{
if(!gtk_tree_model_get_iter(GTK_TREE_MODEL(browser->store),
&iter, p->data))
continue;
gtk_tree_model_get(GTK_TREE_MODEL(browser->store), &iter,
BR_COL_PATH, &q, -1);
printf("%s\n", q);
g_free(q);
}
g_list_foreach(sel, (GFunc)gtk_tree_path_free, NULL);
g_list_free(sel);
}
static void _delete_do(Browser * browser, GList * selection, unsigned long cnt);
void on_edit_delete(GtkMenuItem * menuitem, gpointer data)
{
Browser * browser = data;
GtkWidget * dialog;
unsigned long cnt = 0;
int ret;
GtkTreeIter iter;
GList * selection;
GList * p;
if((selection = _copy_selection(browser)) == NULL)
return;
for(p = selection; p != NULL; p = p->next)
if(!gtk_tree_model_get_iter(GTK_TREE_MODEL(browser->store),
&iter, p->data))
continue;
else
cnt++;
if(cnt == 0)
return;
dialog = gtk_message_dialog_new(GTK_WINDOW(browser->window),
GTK_DIALOG_MODAL | GTK_DIALOG_DESTROY_WITH_PARENT,
GTK_MESSAGE_WARNING, GTK_BUTTONS_YES_NO, "%s%lu%s",
"Are you sure you want to delete ", cnt, " file(s)?");
gtk_window_set_title(GTK_WINDOW(dialog), "Delete file(s)");
ret = gtk_dialog_run(GTK_DIALOG(dialog));
gtk_widget_destroy(GTK_WIDGET(dialog));
if(ret == GTK_RESPONSE_YES)
_delete_do(browser, selection, cnt);
g_list_foreach(selection, (GFunc)gtk_tree_path_free, NULL);
g_list_free(selection);
}
static void _delete_do(Browser * browser, GList * selection, unsigned long cnt)
{
unsigned long i = 1;
char ** argv;
pid_t pid;
GtkTreeIter iter;
GList * p;
gchar * q;
if((pid = fork()) == -1)
{
browser_error(browser, "fork", 0);
return;
}
else if(pid != 0)
return;
if((argv = malloc(sizeof(char*) * (cnt+2))) == NULL)
{
fprintf(stderr, "%s%s\n", "browser: malloc: ", strerror(errno));
exit(2);
}
#ifdef DEBUG
argv[0] = "echo";
#else
argv[0] = "delete";
#endif
for(p = selection; p != NULL && i <= cnt; p = p->next)
{
if(!gtk_tree_model_get_iter(GTK_TREE_MODEL(browser->store),
&iter, p->data))
continue;
gtk_tree_model_get(GTK_TREE_MODEL(browser->store), &iter,
BR_COL_PATH, &q, -1);
argv[i++] = q;
}
if(i != cnt+1)
{
fprintf(stderr, "%s", "browser: Could not delete files\n");
exit(2);
}
argv[i] = NULL;
execvp(argv[0], argv);
fprintf(stderr, "%s%s%s%s\n", "browser: ", argv[0], ": ",
strerror(errno));
exit(2);
}
void on_edit_select_all(GtkMenuItem * menuitem, gpointer data)
{
#if GTK_CHECK_VERSION(2, 6, 0)
Browser * browser = data;
gtk_icon_view_select_all(GTK_ICON_VIEW(browser->iconview));
#endif
}
void on_edit_unselect_all(GtkMenuItem * menuitem, gpointer data)
{
#if GTK_CHECK_VERSION(2, 6, 0)
Browser * browser = data;
gtk_icon_view_unselect_all(GTK_ICON_VIEW(browser->iconview));
#endif
}
static void _preferences_set(Browser * browser);
/* callbacks */
static gboolean _preferences_on_closex(GtkWidget * widget, GdkEvent * event,
gpointer data);
static void _preferences_on_cancel(GtkWidget * widget, gpointer data);
static void _preferences_on_ok(GtkWidget * widget, gpointer data);
void on_edit_preferences(GtkMenuItem * menuitem, gpointer data)
{
Browser * browser = data;
GtkWidget * vbox;
GtkWidget * hbox;
GtkWidget * widget;
GtkSizeGroup * group;
if(browser->pr_window != NULL)
{
gtk_widget_show(browser->pr_window);
return;
}
browser->pr_window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_resizable(GTK_WINDOW(browser->pr_window), FALSE);
gtk_window_set_title(GTK_WINDOW(browser->pr_window),
"File browser preferences");
gtk_window_set_transient_for(GTK_WINDOW(browser->pr_window), GTK_WINDOW(
browser->window));
g_signal_connect(G_OBJECT(browser->pr_window), "delete_event",
G_CALLBACK(_preferences_on_closex), browser);
vbox = gtk_vbox_new(FALSE, 0);
browser->pr_sort = gtk_check_button_new_with_mnemonic(
"Sort _folders first");
gtk_box_pack_start(GTK_BOX(vbox), browser->pr_sort, FALSE, FALSE, 4);
browser->pr_hidden = gtk_check_button_new_with_mnemonic(
"Show _hidden files");
gtk_box_pack_start(GTK_BOX(vbox), browser->pr_hidden, FALSE, FALSE, 4);
/* dialog */
hbox = gtk_hbox_new(FALSE, 0);
group = gtk_size_group_new(GTK_SIZE_GROUP_HORIZONTAL);
widget = gtk_button_new_from_stock(GTK_STOCK_OK);
gtk_size_group_add_widget(group, widget);
g_signal_connect(G_OBJECT(widget), "clicked", G_CALLBACK(
_preferences_on_ok), browser);
gtk_box_pack_end(GTK_BOX(hbox), widget, FALSE, TRUE, 4);
widget = gtk_button_new_from_stock(GTK_STOCK_CANCEL);
gtk_size_group_add_widget(group, widget);
g_signal_connect(G_OBJECT(widget), "clicked", G_CALLBACK(
_preferences_on_cancel), browser);
gtk_box_pack_end(GTK_BOX(hbox), widget, FALSE, TRUE, 4);
gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 4);
gtk_container_add(GTK_CONTAINER(browser->pr_window), vbox);
_preferences_set(browser);
gtk_widget_show_all(browser->pr_window);
}
static void _preferences_set(Browser * browser)
{
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(browser->pr_sort),
browser->prefs.sort_folders_first);
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(browser->pr_hidden),
browser->prefs.show_hidden_files);
}
static gboolean _preferences_on_closex(GtkWidget * widget, GdkEvent * event,
gpointer data)
{
Browser * browser = data;
_preferences_on_cancel(widget, browser);
return TRUE;
}
static void _preferences_on_cancel(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
gtk_widget_hide(browser->pr_window);
_preferences_set(browser);
}
static void _preferences_on_ok(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
gtk_widget_hide(browser->pr_window);
browser->prefs.sort_folders_first = gtk_toggle_button_get_active(
GTK_TOGGLE_BUTTON(browser->pr_sort));
browser->prefs.show_hidden_files = gtk_toggle_button_get_active(
GTK_TOGGLE_BUTTON(browser->pr_hidden));
browser_refresh(browser);
}
/* view menu */
void on_view_home(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
browser_set_location(browser, g_get_home_dir());
}
#if GTK_CHECK_VERSION(2, 6, 0)
void on_view_details(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
browser_set_view(browser, BV_DETAILS);
}
void on_view_icons(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
browser_set_view(browser, BV_ICONS);
}
void on_view_list(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
browser_set_view(browser, BV_LIST);
}
#endif /* GTK_CHECK_VERSION(2, 6, 0) */
/* help menu */
static gboolean _about_on_closex(GtkWidget * widget, GdkEvent * event,
gpointer data);
#if !GTK_CHECK_VERSION(2, 6, 0)
static void _about_on_close(GtkWidget * widget, gpointer data);
static void _about_on_credits(GtkWidget * widget, gpointer data);
static void _about_on_license(GtkWidget * widget, gpointer data);
#endif
void on_help_about(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
static GtkWidget * window = NULL;
char const copyright[] = "Copyright (c) 2006 khorben";
#if GTK_CHECK_VERSION(2, 6, 0)
gsize cnt = 65536;
gchar * buf;
if(window != NULL)
{
gtk_widget_show(window);
return;
}
if((buf = malloc(sizeof(*buf) * cnt)) == NULL)
{
browser_error(browser, "malloc", 0);
return;
}
window = gtk_about_dialog_new();
gtk_window_set_transient_for(GTK_WINDOW(window), GTK_WINDOW(
browser->window));
gtk_about_dialog_set_name(GTK_ABOUT_DIALOG(window), PACKAGE);
gtk_about_dialog_set_version(GTK_ABOUT_DIALOG(window), VERSION);
gtk_about_dialog_set_copyright(GTK_ABOUT_DIALOG(window), copyright);
gtk_about_dialog_set_authors(GTK_ABOUT_DIALOG(window), _authors);
if(g_file_get_contents("/usr/share/common-licenses/GPL-2", &buf, &cnt,
NULL) == TRUE)
gtk_about_dialog_set_license(GTK_ABOUT_DIALOG(window), buf);
else
gtk_about_dialog_set_license(GTK_ABOUT_DIALOG(window),
_license);
free(buf);
g_signal_connect(G_OBJECT(window), "delete_event", G_CALLBACK(
_about_on_closex), window);
g_signal_connect(G_OBJECT(window), "response", G_CALLBACK(
gtk_widget_hide), NULL);
gtk_widget_show(window);
}
#else /* !GTK_CHECK_VERSION(2, 6, 0) */
GtkWidget * vbox;
GtkWidget * hbox;
GtkWidget * button;
if(window != NULL)
{
gtk_widget_show(window);
return;
}
window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_container_set_border_width(GTK_CONTAINER(window), 4);
gtk_window_set_title(GTK_WINDOW(window), "About Browser");
gtk_window_set_transient_for(GTK_WINDOW(window), GTK_WINDOW(
browser->window));
g_signal_connect(G_OBJECT(window), "delete_event", G_CALLBACK(
_about_on_closex), window);
vbox = gtk_vbox_new(FALSE, 2);
gtk_box_pack_start(GTK_BOX(vbox), gtk_label_new(PACKAGE " " VERSION),
FALSE, FALSE, 2);
gtk_box_pack_start(GTK_BOX(vbox), gtk_label_new(copyright), FALSE,
FALSE, 2);
hbox = gtk_hbox_new(TRUE, 4);
button = gtk_button_new_with_mnemonic("C_redits");
g_signal_connect(G_OBJECT(button), "clicked", G_CALLBACK(
_about_on_credits), window);
gtk_box_pack_start(GTK_BOX(hbox), button, FALSE, TRUE, 4);
button = gtk_button_new_with_mnemonic("_License");
g_signal_connect(G_OBJECT(button), "clicked", G_CALLBACK(
_about_on_license), window);
gtk_box_pack_start(GTK_BOX(hbox), button, FALSE, TRUE, 4);
button = gtk_button_new_from_stock(GTK_STOCK_CLOSE);
g_signal_connect(G_OBJECT(button), "clicked", G_CALLBACK(
_about_on_close), window);
gtk_box_pack_end(GTK_BOX(hbox), button, FALSE, TRUE, 4);
gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, FALSE, 4);
gtk_container_add(GTK_CONTAINER(window), vbox);
gtk_widget_show_all(window);
}
#endif /* !GTK_CHECK_VERSION(2, 6, 0) */
static gboolean _about_on_closex(GtkWidget * widget, GdkEvent * event,
gpointer data)
{
gtk_widget_hide(widget);
return TRUE;
}
#if !GTK_CHECK_VERSION(2, 6, 0)
static void _about_on_close(GtkWidget * widget, gpointer data)
{
GtkWidget * window = data;
gtk_widget_hide(window);
}
static void _about_on_credits(GtkWidget * widget, gpointer data)
{
static GtkWidget * window = NULL;
GtkWidget * about = data;
GtkWidget * vbox;
GtkWidget * notebook;
GtkWidget * textview;
GtkTextBuffer * tbuf;
GtkTextIter iter;
GtkWidget * hbox;
size_t i;
if(window != NULL)
{
gtk_widget_show(window);
return;
}
window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_default_size(GTK_WINDOW(window), 200, 200);
gtk_container_set_border_width(GTK_CONTAINER(window), 4);
gtk_window_set_title(GTK_WINDOW(window), "Credits");
gtk_window_set_transient_for(GTK_WINDOW(window), GTK_WINDOW(about));
g_signal_connect(G_OBJECT(window), "delete_event", G_CALLBACK(
_about_on_closex), NULL);
vbox = gtk_vbox_new(FALSE, 0);
textview = gtk_text_view_new();
gtk_text_view_set_editable(GTK_TEXT_VIEW(textview), FALSE);
tbuf = gtk_text_view_get_buffer(GTK_TEXT_VIEW(textview));
gtk_text_buffer_set_text(tbuf, "", 0);
for(i = 0; _authors[i] != NULL; i++)
{
gtk_text_buffer_get_end_iter(tbuf, &iter);
gtk_text_buffer_insert(tbuf, &iter, _authors[i], strlen(
_authors[i]));
}
widget = gtk_scrolled_window_new(NULL, NULL);
gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(widget),
GTK_SHADOW_IN);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(widget),
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
gtk_container_add(GTK_CONTAINER(widget), textview);
notebook = gtk_notebook_new();
gtk_notebook_append_page(GTK_NOTEBOOK(notebook), widget,
gtk_label_new("Written by"));
gtk_box_pack_start(GTK_BOX(vbox), notebook, TRUE, TRUE, 4);
hbox = gtk_hbox_new(FALSE, 0);
widget = gtk_button_new_from_stock(GTK_STOCK_CLOSE);
g_signal_connect(G_OBJECT(widget), "clicked",
G_CALLBACK(_about_on_close), window);
gtk_box_pack_end(GTK_BOX(hbox), widget, FALSE, TRUE, 4);
gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, TRUE, 4);
gtk_container_add(GTK_CONTAINER(window), vbox);
gtk_widget_show_all(window);
}
static void _about_on_license(GtkWidget * widget, gpointer data)
{
static GtkWidget * window = NULL;
GtkWidget * about = data;
GtkWidget * vbox;
GtkWidget * textview;
GtkTextBuffer * tbuf;
GtkWidget * hbox;
if(window != NULL)
{
gtk_widget_show(window);
return;
}
window = gtk_window_new(GTK_WINDOW_TOPLEVEL);
gtk_window_set_default_size(GTK_WINDOW(window), 200, 200);
gtk_container_set_border_width(GTK_CONTAINER(window), 4);
gtk_window_set_title(GTK_WINDOW(window), "License");
gtk_window_set_transient_for(GTK_WINDOW(window), GTK_WINDOW(about));
g_signal_connect(G_OBJECT(window), "delete_event", G_CALLBACK(
_about_on_closex), NULL);
vbox = gtk_vbox_new(FALSE, 0);
textview = gtk_text_view_new();
gtk_text_view_set_editable(GTK_TEXT_VIEW(textview), FALSE);
tbuf = gtk_text_view_get_buffer(GTK_TEXT_VIEW(textview));
gtk_text_buffer_set_text(tbuf, _license, strlen(_license));
widget = gtk_scrolled_window_new(NULL, NULL);
gtk_scrolled_window_set_shadow_type(GTK_SCROLLED_WINDOW(widget),
GTK_SHADOW_IN);
gtk_scrolled_window_set_policy(GTK_SCROLLED_WINDOW(widget),
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
gtk_container_add(GTK_CONTAINER(widget), textview);
gtk_box_pack_start(GTK_BOX(vbox), widget, TRUE, TRUE, 4);
hbox = gtk_hbox_new(FALSE, 0);
widget = gtk_button_new_from_stock(GTK_STOCK_CLOSE);
g_signal_connect(G_OBJECT(widget), "clicked",
G_CALLBACK(_about_on_close), window);
gtk_box_pack_end(GTK_BOX(hbox), widget, FALSE, TRUE, 4);
gtk_box_pack_start(GTK_BOX(vbox), hbox, FALSE, TRUE, 4);
gtk_container_add(GTK_CONTAINER(window), vbox);
gtk_widget_show_all(window);
}
#endif /* !GTK_CHECK_VERSION(2, 6, 0) */
/* toolbar */
void on_back(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
if(browser->current->prev == NULL)
return;
browser->current = g_list_previous(browser->current);
gtk_widget_set_sensitive(GTK_WIDGET(browser->tb_back),
browser->current->prev != NULL);
gtk_widget_set_sensitive(GTK_WIDGET(browser->tb_updir),
strcmp(browser->current->data, "/") != 0);
gtk_widget_set_sensitive(GTK_WIDGET(browser->tb_forward),
TRUE);
browser_refresh(browser);
}
void on_forward(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
if(browser->current->next == NULL)
return;
browser->current = browser->current->next;
gtk_widget_set_sensitive(GTK_WIDGET(browser->tb_back), TRUE);
gtk_widget_set_sensitive(GTK_WIDGET(browser->tb_updir),
strcmp(browser->current->data, "/") != 0);
gtk_widget_set_sensitive(GTK_WIDGET(browser->tb_forward),
browser->current->next != NULL);
browser_refresh(browser);
}
void on_home(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
browser_set_location(browser, g_get_home_dir());
}
void on_properties(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
GList * selection;
if((selection = _copy_selection(browser)) == NULL)
return;
/* FIXME */
g_list_foreach(selection, (GFunc)gtk_tree_path_free, NULL);
g_list_free(selection);
}
void on_refresh(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
browser_refresh(browser);
}
void on_updir(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
char * dir;
browser = data;
dir = g_path_get_dirname(browser->current->data);
browser_set_location(browser, dir);
g_free(dir);
}
#if GTK_CHECK_VERSION(2, 6, 0)
void on_view_as(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
if(browser->iconview == NULL)
browser_set_view(browser, BV_ICONS);
else if(gtk_icon_view_get_orientation(GTK_ICON_VIEW(browser->iconview))
== GTK_ORIENTATION_VERTICAL)
browser_set_view(browser, BV_LIST);
else
browser_set_view(browser, BV_DETAILS);
}
#endif
/* address bar */
void on_path_activate(GtkWidget * widget, gpointer data)
{
Browser * browser = data;
gchar const * p;
widget = gtk_bin_get_child(GTK_BIN(browser->tb_path));
p = gtk_entry_get_text(GTK_ENTRY(widget));
browser_set_location(browser, p);
}
/* view */
/* types */
/* FIXME rather ugly, maybe could go directly in Browser */
typedef struct _IconCallback
{
Browser * browser;
int isdir;
char * path;
} IconCallback;
/* variables */
static IconCallback _icon_cb_data;
static void _default_do(Browser * browser, GtkTreePath * path);
void on_detail_default(GtkTreeView * view, GtkTreePath * path,
GtkTreeViewColumn * column, gpointer data)
{
Browser * browser = data;
_default_do(browser, path);
}
static void _default_do(Browser * browser, GtkTreePath * path)
{
char * location;
GtkTreeIter iter;
gboolean is_dir;
gtk_tree_model_get_iter(GTK_TREE_MODEL(browser->store), &iter, path);
gtk_tree_model_get(GTK_TREE_MODEL(browser->store), &iter, BR_COL_PATH,
&location, BR_COL_IS_DIRECTORY, &is_dir, -1);
if(is_dir)
browser_set_location(browser, location);
else
mime_action(browser->mime, "open", location);
g_free(location);
}
#if GTK_CHECK_VERSION(2, 6, 0)
void on_icon_default(GtkIconView * view,
GtkTreePath * path, gpointer data)
{
Browser * browser = data;
_default_do(browser, path);
}
#endif
void on_filename_edited(GtkCellRendererText * renderer, gchar * arg1,
gchar * arg2, gpointer data)
{
Browser * browser = data;
GtkTreeModel * model;
GtkTreeIter iter;
char * path = NULL;
ssize_t len;
char * q = NULL;
#if GTK_CHECK_VERSION(2, 6, 0)
if(browser->iconview != NULL)
model = gtk_icon_view_get_model(GTK_ICON_VIEW(
browser->iconview));
else
#endif
model = gtk_tree_view_get_model(GTK_TREE_VIEW(browser->detailview));
if(gtk_tree_model_get_iter_from_string(model, &iter, arg1) == TRUE)
{
gtk_tree_model_get(model, &iter, BR_COL_PATH, &path, -1);
if(path != NULL && (len = strrchr(path, '/') - path) > 0
&& strcmp(&path[len+1], arg2) != 0)
q = malloc(len + strlen(arg2) + 2);
}
if(q == NULL)
{
free(path);
return;
}
strncpy(q, path, len);
sprintf(&q[len], "/%s", arg2);
fprintf(stderr, "%s (%s) -> %s, dirlen %u\n", path, arg1, q, len);
if(link(path, q) != 0 || unlink(path) != 0)
browser_error(browser, strerror(errno), 0);
else
gtk_list_store_set(browser->store, &iter, BR_COL_PATH, q,
BR_COL_DISPLAY_NAME, arg2, -1);
free(q);
free(path);
}
static void _popup_mime(Browser * browser, char const * type,
char const * action, char const * label,
GCallback callback, GtkWidget * menu);
static gboolean _popup_show(Browser * browser, GdkEventButton * event,
GtkWidget * menu);
static void _on_icon_delete(GtkWidget * widget, gpointer data);
static void _on_icon_open(GtkWidget * widget, gpointer data);
static void _on_icon_edit(GtkWidget * widget, gpointer data);
static void _on_icon_open_with(GtkWidget * widget, gpointer data);
gboolean on_view_popup(GtkWidget * widget, GdkEventButton * event,
gpointer data)
{
Browser * browser = data;
GtkWidget * menu;
GtkTreePath * path = NULL;
GtkTreeIter iter;
GtkWidget * menuitem;
char * mime = NULL;
if(event->type != GDK_BUTTON_PRESS || event->button != 3)
return FALSE;
menu = gtk_menu_new();
/* FIXME prevents actions to be called but probably leaks memory
g_signal_connect(G_OBJECT(menu), "deactivate", G_CALLBACK(
gtk_widget_destroy), NULL); */
#if GTK_CHECK_VERSION(2, 6, 0)
if(browser->iconview != NULL)
path = gtk_icon_view_get_path_at_pos(GTK_ICON_VIEW(
browser->iconview), (int)event->x,
(int)event->y);
else
#endif
gtk_tree_view_get_path_at_pos(GTK_TREE_VIEW(
browser->detailview), (int)event->x,
(int)event->y, &path, NULL, NULL, NULL);
if(path == NULL)
{
menuitem = gtk_image_menu_item_new_from_stock(
GTK_STOCK_PROPERTIES, NULL);
gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
return _popup_show(browser, event, menu);
}
/* FIXME error checking + sub-functions */
gtk_tree_model_get_iter(GTK_TREE_MODEL(browser->store), &iter, path);
gtk_tree_model_get(GTK_TREE_MODEL(browser->store), &iter,
BR_COL_PATH, &_icon_cb_data.path,
BR_COL_IS_DIRECTORY, &_icon_cb_data.isdir,
BR_COL_MIME_TYPE, &mime, -1);
_icon_cb_data.browser = browser;
if(_icon_cb_data.isdir == TRUE)
{
menuitem = gtk_image_menu_item_new_from_stock(GTK_STOCK_OPEN,
NULL);
g_signal_connect(G_OBJECT(menuitem), "activate", G_CALLBACK(
_on_icon_open), &_icon_cb_data);
gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
}
else /* not a directory */
{
_popup_mime(browser, mime, "open", GTK_STOCK_OPEN,
G_CALLBACK(_on_icon_open), menu);
_popup_mime(browser, mime, "edit",
#if GTK_CHECK_VERSION(2, 6, 0)
GTK_STOCK_EDIT,
#else
"_Edit",
#endif
G_CALLBACK(_on_icon_edit), menu);
menuitem = gtk_menu_item_new_with_mnemonic("Open _with...");
g_signal_connect(G_OBJECT(menuitem), "activate", G_CALLBACK(
_on_icon_open_with), &_icon_cb_data);
gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
menuitem = gtk_separator_menu_item_new();
gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
menuitem = gtk_image_menu_item_new_from_stock(GTK_STOCK_DELETE,
NULL);
g_signal_connect(G_OBJECT(menuitem), "activate", G_CALLBACK(
_on_icon_delete), &_icon_cb_data);
gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
}
g_free(mime);
menuitem = gtk_separator_menu_item_new();
gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
menuitem = gtk_image_menu_item_new_from_stock(
GTK_STOCK_PROPERTIES, NULL);
gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
#if !GTK_CHECK_VERSION(2, 6, 0)
gtk_tree_path_free(path);
#endif
return _popup_show(browser, event, menu);
}
static void _popup_mime(Browser * browser, char const * type,
char const * action, char const * label,
GCallback callback, GtkWidget * menu)
{
GtkWidget * menuitem;
if(mime_get_handler(browser->mime, type, action) == NULL)
return;
if(strncmp(label, "gtk-", 4) == 0)
menuitem = gtk_image_menu_item_new_from_stock(label, NULL);
else
menuitem = gtk_menu_item_new_with_mnemonic(label);
g_signal_connect(G_OBJECT(menuitem), "activate", callback,
&_icon_cb_data);
gtk_menu_shell_append(GTK_MENU_SHELL(menu), menuitem);
}
static gboolean _popup_show(Browser * browser, GdkEventButton * event,
GtkWidget * menu)
{
#if GTK_CHECK_VERSION(2, 6, 0)
if(browser->iconview != NULL)
gtk_menu_attach_to_widget(GTK_MENU(menu), browser->iconview,
NULL);
else
#endif
gtk_menu_attach_to_widget(GTK_MENU(menu), browser->detailview,
NULL);
gtk_widget_show_all(menu);
gtk_menu_popup(GTK_MENU(menu), NULL, NULL, NULL, NULL, 3, event->time);
return TRUE;
}
static void _on_icon_delete(GtkWidget * widget, gpointer data)
{
IconCallback * cb = data;
/* FIXME not selected => cursor */
on_edit_delete(NULL, cb->browser);
}
static void _on_icon_open(GtkWidget * widget, gpointer data)
{
IconCallback * cb = data;
if(cb->isdir)
browser_set_location(cb->browser, cb->path);
else
mime_action(cb->browser->mime, "open", cb->path);
}
static void _on_icon_edit(GtkWidget * widget, gpointer data)
{
IconCallback * cb = data;
mime_action(cb->browser->mime, "edit", cb->path);
}
static void _on_icon_open_with(GtkWidget * widget, gpointer data)
{
IconCallback * cb = data;
GtkWidget * dialog;
char * filename = NULL;
pid_t pid;
dialog = gtk_file_chooser_dialog_new("Open with...",
cb->browser->window, GTK_FILE_CHOOSER_ACTION_OPEN,
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL,
GTK_STOCK_OPEN, GTK_RESPONSE_ACCEPT, NULL);
if(gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT)
filename = gtk_file_chooser_get_filename(GTK_FILE_CHOOSER(
dialog));
gtk_widget_destroy(dialog);
if(filename == NULL)
return;
if((pid = fork()) == -1)
browser_error(cb->browser, "fork", 0);
else if(pid == 0)
{
execlp(filename, filename, cb->path, NULL);
browser_error(NULL, filename, 0);
exit(2);
}
g_free(filename);
}