cdf447647f
prevent conflicts with other tags whilst keeping the same first word as core function names, and update plugins to use them. The old plugin macros are still available, but will be removed after the next release. Compiling with -DGEANY_DISABLE_DEPRECATED in CFLAGS will disable the old macros. git-svn-id: https://geany.svn.sourceforge.net/svnroot/geany/trunk@2195 ea778897-0a13-0410-b9d1-a72fbfd435f5
925 lines
25 KiB
C
925 lines
25 KiB
C
/*
|
|
* filebrowser.c - this file is part of Geany, a fast and lightweight IDE
|
|
*
|
|
* Copyright 2007-2008 Enrico Tröger <enrico(dot)troeger(at)uvena(dot)de>
|
|
* Copyright 2007-2008 Nick Treleaven <nick(dot)treleaven(at)btinternet(dot)com>
|
|
*
|
|
* 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
|
|
* (at your option) any later version.
|
|
*
|
|
* This program is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* 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., 51 Franklin Street, Fifth Floor, Boston,
|
|
* MA 02110-1301, USA.
|
|
*
|
|
* $Id$
|
|
*/
|
|
|
|
/* Sidebar file browser plugin. */
|
|
|
|
#include "geany.h"
|
|
|
|
#include <gdk/gdkkeysyms.h>
|
|
|
|
#include "support.h"
|
|
#include "prefs.h"
|
|
#include "document.h"
|
|
#include "utils.h"
|
|
#include "keybindings.h"
|
|
#include "project.h"
|
|
|
|
#include "plugindata.h"
|
|
#include "pluginmacros.h"
|
|
|
|
|
|
PluginFields *plugin_fields;
|
|
GeanyData *geany_data;
|
|
|
|
|
|
VERSION_CHECK(26)
|
|
|
|
PLUGIN_INFO(_("File Browser"), _("Adds a file browser tab to the sidebar."), VERSION,
|
|
_("The Geany developer team"))
|
|
|
|
|
|
// number of characters to skip the root of an absolute path("c:" or "d:" on Windows)
|
|
#ifdef G_OS_WIN32
|
|
# define ROOT_OFFSET 2
|
|
#else
|
|
# define ROOT_OFFSET 0
|
|
#endif
|
|
|
|
enum
|
|
{
|
|
FILEVIEW_COLUMN_ICON = 0,
|
|
FILEVIEW_COLUMN_NAME,
|
|
FILEVIEW_N_COLUMNS,
|
|
};
|
|
|
|
static gboolean show_hidden_files = FALSE;
|
|
static gboolean hide_object_files = TRUE;
|
|
|
|
static GtkWidget *file_view_vbox;
|
|
static GtkWidget *file_view;
|
|
static GtkListStore *file_store;
|
|
static GtkTreeIter *last_dir_iter = NULL;
|
|
static GtkEntryCompletion *entry_completion = NULL;
|
|
|
|
static GtkWidget *path_entry;
|
|
static gchar *current_dir = NULL; // in locale-encoding
|
|
static gchar *open_cmd; // in locale-encoding
|
|
static gchar *config_file;
|
|
|
|
static struct
|
|
{
|
|
GtkWidget *open;
|
|
GtkWidget *open_external;
|
|
GtkWidget *find_in_files;
|
|
} popup_items;
|
|
|
|
|
|
// Returns: whether name should be hidden.
|
|
static gboolean check_hidden(const gchar *base_name)
|
|
{
|
|
gsize len;
|
|
|
|
if (! NZV(base_name))
|
|
return FALSE;
|
|
|
|
if (base_name[0] == '.')
|
|
return TRUE;
|
|
|
|
len = strlen(base_name);
|
|
if (base_name[len - 1] == '~')
|
|
return TRUE;
|
|
|
|
if (hide_object_files)
|
|
{
|
|
const gchar *exts[] = {".o", ".obj", ".so", ".dll", ".a", ".lib"};
|
|
guint i;
|
|
|
|
for (i = 0; i < G_N_ELEMENTS(exts); i++)
|
|
{
|
|
const gchar *ext = exts[i];
|
|
|
|
if (p_utils->str_equal(&base_name[len - strlen(ext)], ext))
|
|
return TRUE;
|
|
}
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
// name is in locale encoding
|
|
static void add_item(const gchar *name)
|
|
{
|
|
GtkTreeIter iter;
|
|
gchar *fname, *utf8_name;
|
|
gboolean dir;
|
|
|
|
if (! show_hidden_files && check_hidden(name))
|
|
return;
|
|
|
|
fname = g_strconcat(current_dir, G_DIR_SEPARATOR_S, name, NULL);
|
|
dir = g_file_test(fname, G_FILE_TEST_IS_DIR);
|
|
g_free(fname);
|
|
|
|
if (dir)
|
|
{
|
|
if (last_dir_iter == NULL)
|
|
gtk_list_store_prepend(file_store, &iter);
|
|
else
|
|
{
|
|
gtk_list_store_insert_after(file_store, &iter, last_dir_iter);
|
|
gtk_tree_iter_free(last_dir_iter);
|
|
}
|
|
last_dir_iter = gtk_tree_iter_copy(&iter);
|
|
}
|
|
else
|
|
gtk_list_store_append(file_store, &iter);
|
|
|
|
utf8_name = p_utils->get_utf8_from_locale(name);
|
|
|
|
gtk_list_store_set(file_store, &iter,
|
|
FILEVIEW_COLUMN_ICON, (dir) ? GTK_STOCK_DIRECTORY : GTK_STOCK_FILE,
|
|
FILEVIEW_COLUMN_NAME, utf8_name, -1);
|
|
g_free(utf8_name);
|
|
}
|
|
|
|
|
|
static gboolean is_top_level_directory(const gchar *dir)
|
|
{
|
|
g_return_val_if_fail(dir && strlen(dir) > ROOT_OFFSET, FALSE);
|
|
|
|
return (p_utils->str_equal(dir + ROOT_OFFSET, G_DIR_SEPARATOR_S));
|
|
}
|
|
|
|
|
|
// adds ".." to the start of the file list
|
|
static void add_top_level_entry()
|
|
{
|
|
GtkTreeIter iter;
|
|
|
|
if (is_top_level_directory(current_dir))
|
|
return;
|
|
|
|
gtk_list_store_prepend(file_store, &iter);
|
|
last_dir_iter = gtk_tree_iter_copy(&iter);
|
|
|
|
gtk_list_store_set(file_store, &iter,
|
|
FILEVIEW_COLUMN_ICON, GTK_STOCK_DIRECTORY, FILEVIEW_COLUMN_NAME, "..", -1);
|
|
}
|
|
|
|
|
|
static void clear()
|
|
{
|
|
gtk_list_store_clear(file_store);
|
|
|
|
// reset the directory item pointer
|
|
if (last_dir_iter != NULL)
|
|
gtk_tree_iter_free(last_dir_iter);
|
|
last_dir_iter = NULL;
|
|
}
|
|
|
|
|
|
// recreate the tree model from current_dir.
|
|
static void refresh()
|
|
{
|
|
gchar *utf8_dir;
|
|
GSList *list;
|
|
|
|
// don't clear when the new path doesn't exist
|
|
if (! g_file_test(current_dir, G_FILE_TEST_EXISTS))
|
|
return;
|
|
|
|
clear();
|
|
|
|
utf8_dir = p_utils->get_utf8_from_locale(current_dir);
|
|
gtk_entry_set_text(GTK_ENTRY(path_entry), utf8_dir);
|
|
g_free(utf8_dir);
|
|
|
|
list = p_utils->get_file_list(current_dir, NULL, NULL);
|
|
if (list != NULL)
|
|
{
|
|
add_top_level_entry();
|
|
g_slist_foreach(list, (GFunc) add_item, NULL);
|
|
g_slist_foreach(list, (GFunc) g_free, NULL);
|
|
g_slist_free(list);
|
|
}
|
|
gtk_entry_completion_set_model(entry_completion, GTK_TREE_MODEL(file_store));
|
|
}
|
|
|
|
|
|
static void on_go_home()
|
|
{
|
|
setptr(current_dir, g_strdup(g_get_home_dir()));
|
|
refresh();
|
|
}
|
|
|
|
|
|
static gchar *get_default_dir()
|
|
{
|
|
const gchar *dir = NULL;
|
|
|
|
if (project)
|
|
dir = project->base_path;
|
|
if (NZV(dir))
|
|
return p_utils->get_locale_from_utf8(dir);
|
|
|
|
return g_get_current_dir();
|
|
}
|
|
|
|
|
|
static void on_current_path()
|
|
{
|
|
gchar *fname;
|
|
gchar *dir;
|
|
gint idx = p_document->get_cur_idx();
|
|
|
|
if (! DOC_IDX_VALID(idx) || doc_list[idx].file_name == NULL ||
|
|
! g_path_is_absolute(doc_list[idx].file_name))
|
|
{
|
|
setptr(current_dir, get_default_dir());
|
|
refresh();
|
|
return;
|
|
}
|
|
fname = doc_list[idx].file_name;
|
|
fname = p_utils->get_locale_from_utf8(fname);
|
|
dir = g_path_get_dirname(fname);
|
|
g_free(fname);
|
|
|
|
setptr(current_dir, dir);
|
|
refresh();
|
|
}
|
|
|
|
|
|
static void on_go_up()
|
|
{
|
|
// remove the highest directory part (which becomes the basename of current_dir)
|
|
setptr(current_dir, g_path_get_dirname(current_dir));
|
|
refresh();
|
|
}
|
|
|
|
|
|
static gboolean check_single_selection(GtkTreeSelection *treesel)
|
|
{
|
|
if (gtk_tree_selection_count_selected_rows(treesel) == 1)
|
|
return TRUE;
|
|
|
|
p_ui->set_statusbar(FALSE, _("Too many items selected!"));
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
/* Returns: TRUE if at least one of selected_items is a folder. */
|
|
static gboolean is_folder_selected(GList *selected_items)
|
|
{
|
|
GList *item;
|
|
GtkTreeModel *model = GTK_TREE_MODEL(file_store);
|
|
gboolean dir_found = FALSE;
|
|
|
|
for (item = selected_items; item != NULL; item = g_list_next(item))
|
|
{
|
|
gchar *icon;
|
|
GtkTreeIter iter;
|
|
GtkTreePath *treepath;
|
|
|
|
treepath = (GtkTreePath*) item->data;
|
|
gtk_tree_model_get_iter(model, &iter, treepath);
|
|
gtk_tree_model_get(model, &iter, FILEVIEW_COLUMN_ICON, &icon, -1);
|
|
|
|
if (p_utils->str_equal(icon, GTK_STOCK_DIRECTORY))
|
|
{
|
|
dir_found = TRUE;
|
|
g_free(icon);
|
|
break;
|
|
}
|
|
g_free(icon);
|
|
}
|
|
return dir_found;
|
|
}
|
|
|
|
|
|
/* Returns: the full filename in locale encoding. */
|
|
static gchar *get_tree_path_filename(GtkTreePath *treepath)
|
|
{
|
|
GtkTreeModel *model = GTK_TREE_MODEL(file_store);
|
|
GtkTreeIter iter;
|
|
gchar *name, *fname;
|
|
|
|
gtk_tree_model_get_iter(model, &iter, treepath);
|
|
gtk_tree_model_get(model, &iter, FILEVIEW_COLUMN_NAME, &name, -1);
|
|
|
|
if (p_utils->str_equal(name, ".."))
|
|
{
|
|
fname = g_path_get_dirname(current_dir);
|
|
}
|
|
else
|
|
{
|
|
setptr(name, p_utils->get_locale_from_utf8(name));
|
|
fname = g_build_filename(current_dir, name, NULL);
|
|
}
|
|
g_free(name);
|
|
|
|
return fname;
|
|
}
|
|
|
|
|
|
static void open_external(const gchar *fname, gboolean dir_found)
|
|
{
|
|
gchar *cmd;
|
|
gchar *locale_cmd;
|
|
gchar *dir;
|
|
GString *cmd_str = g_string_new(open_cmd);
|
|
GError *error = NULL;
|
|
|
|
if (! dir_found)
|
|
dir = g_path_get_dirname(fname);
|
|
else
|
|
dir = g_strdup(fname);
|
|
|
|
p_utils->string_replace_all(cmd_str, "%f", fname);
|
|
p_utils->string_replace_all(cmd_str, "%d", dir);
|
|
|
|
cmd = g_string_free(cmd_str, FALSE);
|
|
locale_cmd = p_utils->get_locale_from_utf8(cmd);
|
|
if (! g_spawn_command_line_async(locale_cmd, &error))
|
|
{
|
|
gchar *c = strchr(cmd, ' ');
|
|
|
|
if (c != NULL)
|
|
*c = '\0';
|
|
p_ui->set_statusbar(TRUE,
|
|
_("Could not execute configured external command '%s' (%s)."),
|
|
cmd, error->message);
|
|
g_error_free(error);
|
|
}
|
|
g_free(locale_cmd);
|
|
g_free(cmd);
|
|
g_free(dir);
|
|
}
|
|
|
|
|
|
static void on_external_open(GtkMenuItem *menuitem, gpointer user_data)
|
|
{
|
|
GtkTreeSelection *treesel;
|
|
GtkTreeModel *model;
|
|
GList *list;
|
|
gboolean dir_found;
|
|
|
|
treesel = gtk_tree_view_get_selection(GTK_TREE_VIEW(file_view));
|
|
|
|
list = gtk_tree_selection_get_selected_rows(treesel, &model);
|
|
dir_found = is_folder_selected(list);
|
|
|
|
if (! dir_found || check_single_selection(treesel))
|
|
{
|
|
GList *item;
|
|
|
|
for (item = list; item != NULL; item = g_list_next(item))
|
|
{
|
|
GtkTreePath *treepath = item->data;
|
|
gchar *fname = get_tree_path_filename(treepath);
|
|
|
|
open_external(fname, dir_found);
|
|
g_free(fname);
|
|
}
|
|
}
|
|
|
|
g_list_foreach(list, (GFunc) gtk_tree_path_free, NULL);
|
|
g_list_free(list);
|
|
}
|
|
|
|
|
|
/* We use p_document->open_files() as it's more efficient. */
|
|
static void open_selected_files(GList *list)
|
|
{
|
|
GSList *files = NULL;
|
|
GList *item;
|
|
|
|
for (item = list; item != NULL; item = g_list_next(item))
|
|
{
|
|
GtkTreePath *treepath = item->data;
|
|
gchar *fname = get_tree_path_filename(treepath);
|
|
|
|
files = g_slist_append(files, fname);
|
|
}
|
|
p_document->open_files(files, FALSE, NULL, NULL);
|
|
g_slist_foreach(files, (GFunc) g_free, NULL); // free filenames
|
|
g_slist_free(files);
|
|
}
|
|
|
|
|
|
static void open_folder(GtkTreePath *treepath)
|
|
{
|
|
gchar *fname = get_tree_path_filename(treepath);
|
|
|
|
setptr(current_dir, fname);
|
|
refresh();
|
|
}
|
|
|
|
|
|
static void on_open_clicked(GtkMenuItem *menuitem, gpointer user_data)
|
|
{
|
|
GtkTreeSelection *treesel;
|
|
GtkTreeModel *model;
|
|
GList *list;
|
|
gboolean dir_found;
|
|
|
|
treesel = gtk_tree_view_get_selection(GTK_TREE_VIEW(file_view));
|
|
|
|
list = gtk_tree_selection_get_selected_rows(treesel, &model);
|
|
dir_found = is_folder_selected(list);
|
|
|
|
if (dir_found)
|
|
{
|
|
if (check_single_selection(treesel))
|
|
{
|
|
GtkTreePath *treepath = list->data; // first selected item
|
|
|
|
open_folder(treepath);
|
|
}
|
|
}
|
|
else
|
|
open_selected_files(list);
|
|
|
|
g_list_foreach(list, (GFunc) gtk_tree_path_free, NULL);
|
|
g_list_free(list);
|
|
}
|
|
|
|
|
|
static void on_find_in_files(GtkMenuItem *menuitem, gpointer user_data)
|
|
{
|
|
GtkTreeSelection *treesel;
|
|
GtkTreeModel *model;
|
|
GList *list;
|
|
gchar *dir;
|
|
gboolean is_dir = FALSE;
|
|
|
|
treesel = gtk_tree_view_get_selection(GTK_TREE_VIEW(file_view));
|
|
if (! check_single_selection(treesel))
|
|
return;
|
|
|
|
list = gtk_tree_selection_get_selected_rows(treesel, &model);
|
|
is_dir = is_folder_selected(list);
|
|
|
|
if (is_dir)
|
|
{
|
|
GtkTreePath *treepath = list->data; // first selected item
|
|
|
|
dir = get_tree_path_filename(treepath);
|
|
}
|
|
else
|
|
dir = g_strdup(current_dir);
|
|
|
|
g_list_foreach(list, (GFunc) gtk_tree_path_free, NULL);
|
|
g_list_free(list);
|
|
|
|
setptr(dir, p_utils->get_utf8_from_locale(dir));
|
|
p_search->show_find_in_files_dialog(dir);
|
|
g_free(dir);
|
|
}
|
|
|
|
|
|
static void on_hidden_files_clicked(GtkCheckMenuItem *item)
|
|
{
|
|
show_hidden_files = gtk_check_menu_item_get_active(item);
|
|
refresh();
|
|
}
|
|
|
|
|
|
static GtkWidget *create_popup_menu()
|
|
{
|
|
GtkWidget *item, *menu, *image;
|
|
|
|
menu = gtk_menu_new();
|
|
|
|
item = gtk_image_menu_item_new_from_stock(GTK_STOCK_OPEN, NULL);
|
|
gtk_widget_show(item);
|
|
gtk_container_add(GTK_CONTAINER(menu), item);
|
|
g_signal_connect((gpointer) item, "activate",
|
|
G_CALLBACK(on_open_clicked), NULL);
|
|
popup_items.open = item;
|
|
|
|
image = gtk_image_new_from_stock(GTK_STOCK_OPEN, GTK_ICON_SIZE_MENU);
|
|
gtk_widget_show(image);
|
|
item = gtk_image_menu_item_new_with_mnemonic(_("Open _externally"));
|
|
gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item), image);
|
|
gtk_widget_show(item);
|
|
gtk_container_add(GTK_CONTAINER(menu), item);
|
|
g_signal_connect((gpointer) item, "activate",
|
|
G_CALLBACK(on_external_open), NULL);
|
|
popup_items.open_external = item;
|
|
|
|
image = gtk_image_new_from_stock(GTK_STOCK_FIND, GTK_ICON_SIZE_MENU);
|
|
gtk_widget_show(image);
|
|
item = gtk_image_menu_item_new_with_mnemonic(_("_Find in Files"));
|
|
gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item), image);
|
|
gtk_widget_show(item);
|
|
gtk_container_add(GTK_CONTAINER(menu), item);
|
|
g_signal_connect((gpointer) item, "activate", G_CALLBACK(on_find_in_files), NULL);
|
|
popup_items.find_in_files = item;
|
|
|
|
item = gtk_separator_menu_item_new();
|
|
gtk_widget_show(item);
|
|
gtk_container_add(GTK_CONTAINER(menu), item);
|
|
|
|
item = gtk_check_menu_item_new_with_mnemonic(_("Show _Hidden Files"));
|
|
gtk_widget_show(item);
|
|
gtk_container_add(GTK_CONTAINER(menu), item);
|
|
g_signal_connect((gpointer) item, "activate", G_CALLBACK(on_hidden_files_clicked), NULL);
|
|
|
|
item = gtk_separator_menu_item_new();
|
|
gtk_widget_show(item);
|
|
gtk_container_add(GTK_CONTAINER(menu), item);
|
|
|
|
item = gtk_image_menu_item_new_with_mnemonic(_("H_ide Sidebar"));
|
|
gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item),
|
|
gtk_image_new_from_stock("gtk-close", GTK_ICON_SIZE_MENU));
|
|
gtk_widget_show(item);
|
|
gtk_container_add(GTK_CONTAINER(menu), item);
|
|
g_signal_connect_swapped((gpointer) item, "activate",
|
|
G_CALLBACK(p_keybindings->send_command),
|
|
GINT_TO_POINTER(GEANY_KEYS_MENU_SIDEBAR));
|
|
|
|
return menu;
|
|
}
|
|
|
|
|
|
static gboolean on_button_press(GtkWidget *widget, GdkEventButton *event, gpointer user_data)
|
|
{
|
|
if (event->button == 1 && event->type == GDK_2BUTTON_PRESS)
|
|
on_open_clicked(NULL, NULL);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static void update_popup_menu(GtkWidget *popup_menu)
|
|
{
|
|
GtkTreeSelection *treesel = gtk_tree_view_get_selection(GTK_TREE_VIEW(file_view));
|
|
gboolean have_sel = (gtk_tree_selection_count_selected_rows(treesel) > 0);
|
|
gboolean multi_sel = (gtk_tree_selection_count_selected_rows(treesel) > 1);
|
|
|
|
gtk_widget_set_sensitive(popup_items.open, have_sel);
|
|
gtk_widget_set_sensitive(popup_items.open_external, have_sel);
|
|
gtk_widget_set_sensitive(popup_items.find_in_files, have_sel && ! multi_sel);
|
|
}
|
|
|
|
|
|
// delay updating popup menu until the selection has been set
|
|
static gboolean on_button_release(GtkWidget *widget, GdkEventButton *event, gpointer user_data)
|
|
{
|
|
if (event->button == 3)
|
|
{
|
|
static GtkWidget *popup_menu = NULL;
|
|
|
|
if (popup_menu == NULL)
|
|
popup_menu = create_popup_menu();
|
|
|
|
update_popup_menu(popup_menu);
|
|
|
|
gtk_menu_popup(GTK_MENU(popup_menu), NULL, NULL, NULL, NULL,
|
|
event->button, event->time);
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static gboolean on_key_press(GtkWidget *widget, GdkEventKey *event, gpointer data)
|
|
{
|
|
if (event->keyval == GDK_Return
|
|
|| event->keyval == GDK_ISO_Enter
|
|
|| event->keyval == GDK_KP_Enter
|
|
|| event->keyval == GDK_space)
|
|
on_open_clicked(NULL, NULL);
|
|
|
|
if ((event->keyval == GDK_Up ||
|
|
event->keyval == GDK_KP_Up) &&
|
|
(event->state & GDK_MOD1_MASK)) // FIXME: Alt-Up doesn't seem to work!
|
|
on_go_up();
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static void on_path_entry_activate(GtkEntry *entry, gpointer user_data)
|
|
{
|
|
gchar *new_dir = (gchar*) gtk_entry_get_text(entry);
|
|
|
|
if (NZV(new_dir))
|
|
{
|
|
if (g_str_has_suffix(new_dir, ".."))
|
|
{
|
|
on_go_up();
|
|
return;
|
|
}
|
|
new_dir = p_utils->get_locale_from_utf8(new_dir);
|
|
}
|
|
else
|
|
new_dir = g_strdup(g_get_home_dir());
|
|
|
|
setptr(current_dir, new_dir);
|
|
refresh();
|
|
}
|
|
|
|
|
|
static void prepare_file_view()
|
|
{
|
|
GtkCellRenderer *text_renderer, *icon_renderer;
|
|
GtkTreeViewColumn *column;
|
|
GtkTreeSelection *select;
|
|
PangoFontDescription *pfd;
|
|
|
|
file_store = gtk_list_store_new(FILEVIEW_N_COLUMNS, G_TYPE_STRING, G_TYPE_STRING);
|
|
|
|
gtk_tree_view_set_model(GTK_TREE_VIEW(file_view), GTK_TREE_MODEL(file_store));
|
|
|
|
icon_renderer = gtk_cell_renderer_pixbuf_new();
|
|
text_renderer = gtk_cell_renderer_text_new();
|
|
column = gtk_tree_view_column_new();
|
|
gtk_tree_view_column_pack_start(column, icon_renderer, FALSE);
|
|
gtk_tree_view_column_set_attributes(column, icon_renderer, "stock-id", FILEVIEW_COLUMN_ICON, NULL);
|
|
gtk_tree_view_column_pack_start(column, text_renderer, TRUE);
|
|
gtk_tree_view_column_set_attributes(column, text_renderer, "text", FILEVIEW_COLUMN_NAME, NULL);
|
|
gtk_tree_view_append_column(GTK_TREE_VIEW(file_view), column);
|
|
gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(file_view), FALSE);
|
|
|
|
gtk_tree_view_set_enable_search(GTK_TREE_VIEW(file_view), TRUE);
|
|
gtk_tree_view_set_search_column(GTK_TREE_VIEW(file_view), FILEVIEW_COLUMN_NAME);
|
|
|
|
pfd = pango_font_description_from_string(prefs->tagbar_font);
|
|
gtk_widget_modify_font(file_view, pfd);
|
|
pango_font_description_free(pfd);
|
|
|
|
// selection handling
|
|
select = gtk_tree_view_get_selection(GTK_TREE_VIEW(file_view));
|
|
gtk_tree_selection_set_mode(select, GTK_SELECTION_MULTIPLE);
|
|
|
|
g_signal_connect(G_OBJECT(file_view), "realize", G_CALLBACK(on_current_path), NULL);
|
|
g_signal_connect(G_OBJECT(file_view), "button-press-event",
|
|
G_CALLBACK(on_button_press), NULL);
|
|
g_signal_connect(G_OBJECT(file_view), "button-release-event",
|
|
G_CALLBACK(on_button_release), NULL);
|
|
g_signal_connect(G_OBJECT(file_view), "key-press-event",
|
|
G_CALLBACK(on_key_press), NULL);
|
|
}
|
|
|
|
|
|
static GtkWidget *make_toolbar()
|
|
{
|
|
GtkWidget *wid, *toolbar;
|
|
GtkTooltips *tooltips = GTK_TOOLTIPS(p_support->lookup_widget(
|
|
app->window, "tooltips"));
|
|
|
|
toolbar = gtk_toolbar_new();
|
|
gtk_toolbar_set_icon_size(GTK_TOOLBAR(toolbar), GTK_ICON_SIZE_MENU);
|
|
gtk_toolbar_set_style(GTK_TOOLBAR(toolbar), GTK_TOOLBAR_ICONS);
|
|
|
|
wid = (GtkWidget *) gtk_tool_button_new_from_stock(GTK_STOCK_GO_UP);
|
|
gtk_tool_item_set_tooltip(GTK_TOOL_ITEM(wid), tooltips,
|
|
_("Up"), NULL);
|
|
g_signal_connect(G_OBJECT(wid), "clicked", G_CALLBACK(on_go_up), NULL);
|
|
gtk_container_add(GTK_CONTAINER(toolbar), wid);
|
|
|
|
wid = (GtkWidget *) gtk_tool_button_new_from_stock(GTK_STOCK_REFRESH);
|
|
gtk_tool_item_set_tooltip(GTK_TOOL_ITEM(wid), tooltips,
|
|
_("Refresh"), NULL);
|
|
g_signal_connect(G_OBJECT(wid), "clicked", G_CALLBACK(refresh), NULL);
|
|
gtk_container_add(GTK_CONTAINER(toolbar), wid);
|
|
|
|
wid = (GtkWidget *) gtk_tool_button_new_from_stock(GTK_STOCK_HOME);
|
|
gtk_tool_item_set_tooltip(GTK_TOOL_ITEM(wid), tooltips,
|
|
_("Home"), NULL);
|
|
g_signal_connect(G_OBJECT(wid), "clicked", G_CALLBACK(on_go_home), NULL);
|
|
gtk_container_add(GTK_CONTAINER(toolbar), wid);
|
|
|
|
wid = (GtkWidget *) gtk_tool_button_new_from_stock(GTK_STOCK_JUMP_TO);
|
|
gtk_tool_item_set_tooltip(GTK_TOOL_ITEM(wid), tooltips,
|
|
_("Set path from document"), NULL);
|
|
g_signal_connect(G_OBJECT(wid), "clicked", G_CALLBACK(on_current_path), NULL);
|
|
gtk_container_add(GTK_CONTAINER(toolbar), wid);
|
|
|
|
return toolbar;
|
|
}
|
|
|
|
|
|
static gboolean completion_match_func(GtkEntryCompletion *completion, const gchar *key,
|
|
GtkTreeIter *iter, gpointer user_data)
|
|
{
|
|
gchar *str, *icon;
|
|
gboolean result = FALSE;
|
|
|
|
gtk_tree_model_get(GTK_TREE_MODEL(file_store), iter,
|
|
FILEVIEW_COLUMN_ICON, &icon, FILEVIEW_COLUMN_NAME, &str, -1);
|
|
|
|
if (str != NULL && icon != NULL && p_utils->str_equal(icon, GTK_STOCK_DIRECTORY) &&
|
|
! g_str_has_suffix(key, G_DIR_SEPARATOR_S))
|
|
{
|
|
// key is something like "/tmp/te" and str is a filename like "test",
|
|
// so strip the path from key to make them comparable
|
|
gchar *base_name = g_path_get_basename(key);
|
|
gchar *str_lowered = g_utf8_strdown(str, -1);
|
|
result = g_str_has_prefix(str_lowered, base_name);
|
|
g_free(base_name);
|
|
g_free(str_lowered);
|
|
}
|
|
g_free(str);
|
|
g_free(icon);
|
|
|
|
return result;
|
|
}
|
|
|
|
|
|
static gboolean completion_match_selected(GtkEntryCompletion *widget, GtkTreeModel *model,
|
|
GtkTreeIter *iter, gpointer user_data)
|
|
{
|
|
gchar *str;
|
|
gtk_tree_model_get(model, iter, FILEVIEW_COLUMN_NAME, &str, -1);
|
|
if (str != NULL)
|
|
{
|
|
gchar *text = g_strconcat(current_dir, G_DIR_SEPARATOR_S, str, NULL);
|
|
gtk_entry_set_text(GTK_ENTRY(path_entry), text);
|
|
gtk_editable_set_position(GTK_EDITABLE(path_entry), -1);
|
|
// force change of directory when completion is done
|
|
on_path_entry_activate(GTK_ENTRY(path_entry), NULL);
|
|
g_free(text);
|
|
}
|
|
g_free(str);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static void completion_create()
|
|
{
|
|
entry_completion = gtk_entry_completion_new();
|
|
|
|
gtk_entry_completion_set_inline_completion(entry_completion, FALSE);
|
|
gtk_entry_completion_set_popup_completion(entry_completion, TRUE);
|
|
gtk_entry_completion_set_text_column(entry_completion, FILEVIEW_COLUMN_NAME);
|
|
gtk_entry_completion_set_match_func(entry_completion, completion_match_func, NULL, NULL);
|
|
|
|
g_signal_connect(entry_completion, "match-selected",
|
|
G_CALLBACK(completion_match_selected), NULL);
|
|
|
|
gtk_entry_set_completion(GTK_ENTRY(path_entry), entry_completion);
|
|
}
|
|
|
|
|
|
#define CHECK_READ_SETTING(var, error, tmp) \
|
|
if ((error) != NULL) \
|
|
{ \
|
|
g_error_free((error)); \
|
|
(error) = NULL; \
|
|
} \
|
|
else \
|
|
(var) = (tmp);
|
|
|
|
void init(GeanyData *data)
|
|
{
|
|
GtkWidget *scrollwin, *toolbar;
|
|
GKeyFile *config = g_key_file_new();
|
|
GError *error = NULL;
|
|
gboolean tmp;
|
|
|
|
file_view_vbox = gtk_vbox_new(FALSE, 0);
|
|
toolbar = make_toolbar();
|
|
gtk_box_pack_start(GTK_BOX(file_view_vbox), toolbar, FALSE, FALSE, 0);
|
|
|
|
path_entry = gtk_entry_new();
|
|
gtk_box_pack_start(GTK_BOX(file_view_vbox), path_entry, FALSE, FALSE, 2);
|
|
g_signal_connect(G_OBJECT(path_entry), "activate", G_CALLBACK(on_path_entry_activate), NULL);
|
|
|
|
file_view = gtk_tree_view_new();
|
|
prepare_file_view();
|
|
completion_create();
|
|
|
|
scrollwin = gtk_scrolled_window_new(NULL, NULL);
|
|
gtk_scrolled_window_set_policy(
|
|
GTK_SCROLLED_WINDOW(scrollwin),
|
|
GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC);
|
|
gtk_container_add(GTK_CONTAINER(scrollwin), file_view);
|
|
gtk_container_add(GTK_CONTAINER(file_view_vbox), scrollwin);
|
|
|
|
gtk_widget_show_all(file_view_vbox);
|
|
gtk_notebook_append_page(GTK_NOTEBOOK(app->treeview_notebook), file_view_vbox,
|
|
gtk_label_new(_("Files")));
|
|
|
|
config_file = g_strconcat(app->configdir, G_DIR_SEPARATOR_S, "plugins", G_DIR_SEPARATOR_S,
|
|
"filebrowser", G_DIR_SEPARATOR_S, "filebrowser.conf", NULL);
|
|
g_key_file_load_from_file(config, config_file, G_KEY_FILE_NONE, NULL);
|
|
open_cmd = g_key_file_get_string(config, "filebrowser", "open_command", &error);
|
|
if (error != NULL)
|
|
{
|
|
open_cmd = g_strdup("nautilus \"%d\"");
|
|
g_error_free(error);
|
|
error = NULL;
|
|
}
|
|
tmp = g_key_file_get_boolean(config, "filebrowser", "show_hidden_files", &error);
|
|
CHECK_READ_SETTING(show_hidden_files, error, tmp);
|
|
tmp = g_key_file_get_boolean(config, "filebrowser", "hide_object_files", &error);
|
|
CHECK_READ_SETTING(hide_object_files, error, tmp);
|
|
|
|
g_key_file_free(config);
|
|
}
|
|
|
|
|
|
void configure(GtkWidget *parent)
|
|
{
|
|
GtkWidget *dialog, *label, *entry, *checkbox_of, *checkbox_hf, *vbox;
|
|
GtkTooltips *tooltips = gtk_tooltips_new();
|
|
|
|
dialog = gtk_dialog_new_with_buttons(_("File Browser"),
|
|
GTK_WINDOW(parent), GTK_DIALOG_DESTROY_WITH_PARENT,
|
|
GTK_STOCK_CANCEL, GTK_RESPONSE_CANCEL, GTK_STOCK_OK, GTK_RESPONSE_ACCEPT, NULL);
|
|
vbox = p_ui->dialog_vbox_new(GTK_DIALOG(dialog));
|
|
gtk_widget_set_name(dialog, "GeanyDialog");
|
|
gtk_box_set_spacing(GTK_BOX(vbox), 6);
|
|
|
|
label = gtk_label_new("External open command:");
|
|
gtk_misc_set_alignment(GTK_MISC(label), 0, 0.5);
|
|
gtk_container_add(GTK_CONTAINER(vbox), label);
|
|
|
|
entry = gtk_entry_new();
|
|
gtk_widget_show(entry);
|
|
if (open_cmd != NULL)
|
|
gtk_entry_set_text(GTK_ENTRY(entry), open_cmd);
|
|
gtk_tooltips_set_tip(tooltips, entry,
|
|
_("The command to execute when using \"Open with\". You can use %f and %d wildcards.\n"
|
|
"%f will be replaced with the filename including full path\n"
|
|
"%d will be replaced with the path name of the selected file without the filename"),
|
|
NULL);
|
|
gtk_container_add(GTK_CONTAINER(vbox), entry);
|
|
|
|
checkbox_hf = gtk_check_button_new_with_label(_("Show hidden files"));
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(checkbox_hf), FALSE);
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbox_hf), show_hidden_files);
|
|
gtk_box_pack_start(GTK_BOX(vbox), checkbox_hf, FALSE, FALSE, 5);
|
|
|
|
checkbox_of = gtk_check_button_new_with_label(_("Hide object files"));
|
|
gtk_button_set_focus_on_click(GTK_BUTTON(checkbox_of), FALSE);
|
|
gtk_toggle_button_set_active(GTK_TOGGLE_BUTTON(checkbox_of), hide_object_files);
|
|
gtk_tooltips_set_tip(tooltips, checkbox_of,
|
|
_("Don't show generated object files in the file browser, this includes "
|
|
"*.o, *.obj. *.so, *.dll, *.a, *.lib"),
|
|
NULL);
|
|
gtk_box_pack_start(GTK_BOX(vbox), checkbox_of, FALSE, FALSE, 5);
|
|
|
|
|
|
gtk_widget_show_all(vbox);
|
|
|
|
// run the dialog and check for the response code
|
|
if (gtk_dialog_run(GTK_DIALOG(dialog)) == GTK_RESPONSE_ACCEPT)
|
|
{
|
|
GKeyFile *config = g_key_file_new();
|
|
gchar *data;
|
|
gchar *config_dir = g_path_get_dirname(config_file);
|
|
|
|
g_free(open_cmd);
|
|
open_cmd = g_strdup(gtk_entry_get_text(GTK_ENTRY(entry)));
|
|
show_hidden_files = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(checkbox_hf));
|
|
hide_object_files = gtk_toggle_button_get_active(GTK_TOGGLE_BUTTON(checkbox_of));
|
|
|
|
g_key_file_load_from_file(config, config_file, G_KEY_FILE_NONE, NULL);
|
|
|
|
g_key_file_set_string(config, "filebrowser", "open_command", open_cmd);
|
|
g_key_file_set_boolean(config, "filebrowser", "show_hidden_files", show_hidden_files);
|
|
g_key_file_set_boolean(config, "filebrowser", "hide_object_files", hide_object_files);
|
|
|
|
if (! g_file_test(config_dir, G_FILE_TEST_IS_DIR) && p_utils->mkdir(config_dir, TRUE) != 0)
|
|
{
|
|
p_dialogs->show_msgbox(GTK_MESSAGE_ERROR,
|
|
_("Plugin configuration directory could not be created."));
|
|
}
|
|
else
|
|
{
|
|
// write config to file
|
|
data = g_key_file_to_data(config, NULL, NULL);
|
|
p_utils->write_file(config_file, data);
|
|
g_free(data);
|
|
}
|
|
|
|
// apply the changes
|
|
refresh();
|
|
|
|
g_free(config_dir);
|
|
g_key_file_free(config);
|
|
}
|
|
gtk_widget_destroy(dialog);
|
|
}
|
|
|
|
|
|
void cleanup()
|
|
{
|
|
g_free(config_file);
|
|
g_free(open_cmd);
|
|
gtk_widget_destroy(file_view_vbox);
|
|
g_object_unref(G_OBJECT(entry_completion));
|
|
}
|