/* * sidebar.c - this file is part of Geany, a fast and lightweight IDE * * Copyright 2005-2011 Enrico Tröger * Copyright 2006-2011 Nick Treleaven * * 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 related code for the Symbol list and Open files GtkTreeViews. */ #include #include "geany.h" #include "support.h" #include "callbacks.h" #include "sidebar.h" #include "document.h" #include "editor.h" #include "documentprivate.h" #include "filetypes.h" #include "utils.h" #include "ui_utils.h" #include "symbols.h" #include "navqueue.h" #include "project.h" #include "stash.h" #include "keyfile.h" #include "sciwrappers.h" #include "search.h" #include SidebarTreeviews tv = {NULL, NULL, NULL}; /* while typeahead searching, editor should not get focus */ static gboolean may_steal_focus = FALSE; static struct { GtkWidget *close; GtkWidget *save; GtkWidget *reload; GtkWidget *show_paths; GtkWidget *find_in_files; } doc_items = {NULL, NULL, NULL, NULL, NULL}; enum { TREEVIEW_SYMBOL = 0, TREEVIEW_OPENFILES }; enum { OPENFILES_ACTION_REMOVE = 0, OPENFILES_ACTION_SAVE, OPENFILES_ACTION_RELOAD }; /* documents tree model columns */ enum { DOCUMENTS_ICON, DOCUMENTS_SHORTNAME, /* dirname for parents, basename for children */ DOCUMENTS_DOCUMENT, DOCUMENTS_COLOR, DOCUMENTS_FILENAME /* full filename */ }; static GtkTreeStore *store_openfiles; static GtkWidget *openfiles_popup_menu; static gboolean documents_show_paths; static GtkWidget *tag_window; /* scrolled window that holds the symbol list GtkTreeView */ /* callback prototypes */ static void on_openfiles_document_action(GtkMenuItem *menuitem, gpointer user_data); static gboolean sidebar_button_press_cb(GtkWidget *widget, GdkEventButton *event, gpointer user_data); static gboolean sidebar_key_press_cb(GtkWidget *widget, GdkEventKey *event, gpointer user_data); static void on_list_document_activate(GtkCheckMenuItem *item, gpointer user_data); static void on_list_symbol_activate(GtkCheckMenuItem *item, gpointer user_data); static void documents_menu_update(GtkTreeSelection *selection); static void sidebar_tabs_show_hide(GtkNotebook *notebook, GtkWidget *child, guint page_num, gpointer data); /* the prepare_* functions are document-related, but I think they fit better here than in document.c */ static void prepare_taglist(GtkWidget *tree, GtkTreeStore *store) { GtkCellRenderer *text_renderer, *icon_renderer; GtkTreeViewColumn *column; GtkTreeSelection *selection; text_renderer = gtk_cell_renderer_text_new(); icon_renderer = gtk_cell_renderer_pixbuf_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, "pixbuf", SYMBOLS_COLUMN_ICON, NULL); g_object_set(icon_renderer, "xalign", 0.0, NULL); gtk_tree_view_column_pack_start(column, text_renderer, TRUE); gtk_tree_view_column_set_attributes(column, text_renderer, "text", SYMBOLS_COLUMN_NAME, NULL); g_object_set(text_renderer, "yalign", 0.5, NULL); gtk_tree_view_column_set_title(column, _("Symbols")); gtk_tree_view_append_column(GTK_TREE_VIEW(tree), column); gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(tree), FALSE); ui_widget_modify_font_from_string(tree, interface_prefs.tagbar_font); gtk_tree_view_set_model(GTK_TREE_VIEW(tree), GTK_TREE_MODEL(store)); g_object_unref(store); g_signal_connect(tree, "button-press-event", G_CALLBACK(sidebar_button_press_cb), NULL); g_signal_connect(tree, "key-press-event", G_CALLBACK(sidebar_key_press_cb), NULL); gtk_tree_view_set_show_expanders(GTK_TREE_VIEW(tree), interface_prefs.show_symbol_list_expanders); if (! interface_prefs.show_symbol_list_expanders) gtk_tree_view_set_level_indentation(GTK_TREE_VIEW(tree), 10); /* Tooltips */ gtk_tree_view_set_tooltip_column(GTK_TREE_VIEW(tree), SYMBOLS_COLUMN_TOOLTIP); /* selection handling */ selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tree)); gtk_tree_selection_set_mode(selection, GTK_SELECTION_SINGLE); /* callback for changed selection not necessary, will be handled by button-press-event */ } static gboolean on_default_tag_tree_button_press_event(GtkWidget *widget, GdkEventButton *event, gpointer user_data) { if (event->button == 3) { gtk_menu_popup(GTK_MENU(tv.popup_taglist), NULL, NULL, NULL, NULL, event->button, event->time); return TRUE; } return FALSE; } static void create_default_tag_tree(void) { GtkScrolledWindow *scrolled_window = GTK_SCROLLED_WINDOW(tag_window); GtkWidget *label; /* default_tag_tree is a GtkViewPort with a GtkLabel inside it */ tv.default_tag_tree = gtk_viewport_new( gtk_scrolled_window_get_hadjustment(scrolled_window), gtk_scrolled_window_get_vadjustment(scrolled_window)); label = gtk_label_new(_("No tags found")); gtk_misc_set_alignment(GTK_MISC(label), 0.1f, 0.01f); gtk_container_add(GTK_CONTAINER(tv.default_tag_tree), label); gtk_widget_show_all(tv.default_tag_tree); g_signal_connect(tv.default_tag_tree, "button-press-event", G_CALLBACK(on_default_tag_tree_button_press_event), NULL); g_object_ref((gpointer)tv.default_tag_tree); /* to hold it after removing */ } /* update = rescan the tags for doc->filename */ void sidebar_update_tag_list(GeanyDocument *doc, gboolean update) { GtkWidget *child = gtk_bin_get_child(GTK_BIN(tag_window)); /* changes the tree view to the given one, trying not to do useless changes */ #define CHANGE_TREE(new_child) \ G_STMT_START { \ /* only change the tag tree if it's actually not the same (to avoid flickering) and if * it's the one of the current document (to avoid problems when e.g. reloading * configuration files */ \ if (child != new_child && doc == document_get_current()) \ { \ if (child) \ gtk_container_remove(GTK_CONTAINER(tag_window), child); \ gtk_container_add(GTK_CONTAINER(tag_window), new_child); \ } \ } G_STMT_END if (tv.default_tag_tree == NULL) create_default_tag_tree(); /* show default empty tag tree if there are no tags */ if (doc == NULL || doc->file_type == NULL || ! filetype_has_tags(doc->file_type)) { CHANGE_TREE(tv.default_tag_tree); return; } if (update) { /* updating the tag list in the left tag window */ if (doc->priv->tag_tree == NULL) { doc->priv->tag_store = gtk_tree_store_new( SYMBOLS_N_COLUMNS, GDK_TYPE_PIXBUF, G_TYPE_STRING, TM_TYPE_TAG, G_TYPE_STRING, G_TYPE_BOOLEAN); doc->priv->tag_tree = gtk_tree_view_new(); prepare_taglist(doc->priv->tag_tree, doc->priv->tag_store); gtk_widget_show(doc->priv->tag_tree); g_object_ref((gpointer)doc->priv->tag_tree); /* to hold it after removing */ } doc->has_tags = symbols_recreate_tag_list(doc, SYMBOLS_SORT_USE_PREVIOUS); } if (doc->has_tags) { CHANGE_TREE(doc->priv->tag_tree); } else { CHANGE_TREE(tv.default_tag_tree); } #undef CHANGE_TREE } /* cleverly sorts documents by their short name */ static gint documents_sort_func(GtkTreeModel *model, GtkTreeIter *iter_a, GtkTreeIter *iter_b, gpointer data) { gchar *key_a, *key_b; gchar *name_a, *name_b; gint cmp; gtk_tree_model_get(model, iter_a, DOCUMENTS_SHORTNAME, &name_a, -1); key_a = g_utf8_collate_key_for_filename(name_a, -1); g_free(name_a); gtk_tree_model_get(model, iter_b, DOCUMENTS_SHORTNAME, &name_b, -1); key_b = g_utf8_collate_key_for_filename(name_b, -1); g_free(name_b); cmp = strcmp(key_a, key_b); g_free(key_b); g_free(key_a); return cmp; } /* does some preparing things to the open files list widget */ static void prepare_openfiles(void) { GtkCellRenderer *icon_renderer; GtkCellRenderer *text_renderer; GtkTreeViewColumn *column; GtkTreeSelection *selection; GtkTreeSortable *sortable; tv.tree_openfiles = ui_lookup_widget(main_widgets.window, "treeview6"); /* store the icon and the short filename to show, and the index as reference, * the colour (black/red/green) and the full name for the tooltip */ store_openfiles = gtk_tree_store_new(5, GDK_TYPE_PIXBUF, G_TYPE_STRING, G_TYPE_POINTER, GDK_TYPE_COLOR, G_TYPE_STRING); gtk_tree_view_set_model(GTK_TREE_VIEW(tv.tree_openfiles), GTK_TREE_MODEL(store_openfiles)); /* set policy settings for the scolledwindow around the treeview again, because glade * doesn't keep the settings */ gtk_scrolled_window_set_policy( GTK_SCROLLED_WINDOW(ui_lookup_widget(main_widgets.window, "scrolledwindow7")), GTK_POLICY_AUTOMATIC, GTK_POLICY_AUTOMATIC); icon_renderer = gtk_cell_renderer_pixbuf_new(); text_renderer = gtk_cell_renderer_text_new(); g_object_set(text_renderer, "ellipsize", PANGO_ELLIPSIZE_MIDDLE, NULL); 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, "pixbuf", DOCUMENTS_ICON, NULL); gtk_tree_view_column_pack_start(column, text_renderer, TRUE); gtk_tree_view_column_set_attributes(column, text_renderer, "text", DOCUMENTS_SHORTNAME, "foreground-gdk", DOCUMENTS_COLOR, NULL); gtk_tree_view_append_column(GTK_TREE_VIEW(tv.tree_openfiles), column); gtk_tree_view_set_headers_visible(GTK_TREE_VIEW(tv.tree_openfiles), FALSE); gtk_tree_view_set_search_column(GTK_TREE_VIEW(tv.tree_openfiles), DOCUMENTS_SHORTNAME); /* sort opened filenames in the store_openfiles treeview */ sortable = GTK_TREE_SORTABLE(GTK_TREE_MODEL(store_openfiles)); gtk_tree_sortable_set_sort_func(sortable, DOCUMENTS_SHORTNAME, documents_sort_func, NULL, NULL); gtk_tree_sortable_set_sort_column_id(sortable, DOCUMENTS_SHORTNAME, GTK_SORT_ASCENDING); ui_widget_modify_font_from_string(tv.tree_openfiles, interface_prefs.tagbar_font); /* tooltips */ gtk_tree_view_set_tooltip_column(GTK_TREE_VIEW(tv.tree_openfiles), DOCUMENTS_FILENAME); /* selection handling */ selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tv.tree_openfiles)); gtk_tree_selection_set_mode(selection, GTK_SELECTION_SINGLE); g_object_unref(store_openfiles); g_signal_connect(GTK_TREE_VIEW(tv.tree_openfiles), "button-press-event", G_CALLBACK(sidebar_button_press_cb), NULL); g_signal_connect(GTK_TREE_VIEW(tv.tree_openfiles), "key-press-event", G_CALLBACK(sidebar_key_press_cb), NULL); } /* iter should be toplevel */ static gboolean find_tree_iter_dir(GtkTreeIter *iter, const gchar *dir) { GeanyDocument *doc; gchar *name; gboolean result; if (utils_str_equal(dir, ".")) dir = GEANY_STRING_UNTITLED; gtk_tree_model_get(GTK_TREE_MODEL(store_openfiles), iter, DOCUMENTS_DOCUMENT, &doc, -1); g_return_val_if_fail(!doc, FALSE); gtk_tree_model_get(GTK_TREE_MODEL(store_openfiles), iter, DOCUMENTS_SHORTNAME, &name, -1); result = utils_str_equal(name, dir); g_free(name); return result; } static gchar *get_doc_folder(const gchar *path) { gchar *tmp_dirname = g_strdup(path); gchar *project_base_path; gchar *dirname = NULL; const gchar *home_dir = g_get_home_dir(); const gchar *rest; /* replace the project base path with the project name */ project_base_path = project_get_base_path(); if (project_base_path != NULL) { gsize len = strlen(project_base_path); if (project_base_path[len-1] == G_DIR_SEPARATOR) project_base_path[len-1] = '\0'; /* check whether the dir name matches or uses the project base path */ if (g_str_has_prefix(tmp_dirname, project_base_path)) { rest = tmp_dirname + len; if (*rest == G_DIR_SEPARATOR || *rest == '\0') { dirname = g_strdup_printf("%s%s", app->project->name, rest); } } g_free(project_base_path); } if (dirname == NULL) { dirname = tmp_dirname; /* If matches home dir, replace with tilde */ if (home_dir && *home_dir != 0 && g_str_has_prefix(dirname, home_dir)) { rest = dirname + strlen(home_dir); if (*rest == G_DIR_SEPARATOR || *rest == '\0') { dirname = g_strdup_printf("~%s", rest); g_free(tmp_dirname); } } } else g_free(tmp_dirname); return dirname; } static GtkTreeIter *get_doc_parent(GeanyDocument *doc) { gchar *path; gchar *dirname = NULL; static GtkTreeIter parent; GtkTreeModel *model = GTK_TREE_MODEL(store_openfiles); static GdkPixbuf *dir_icon = NULL; if (!documents_show_paths) return NULL; path = g_path_get_dirname(DOC_FILENAME(doc)); dirname = get_doc_folder(path); if (gtk_tree_model_get_iter_first(model, &parent)) { do { if (find_tree_iter_dir(&parent, dirname)) { g_free(dirname); g_free(path); return &parent; } } while (gtk_tree_model_iter_next(model, &parent)); } /* no match, add dir parent */ if (!dir_icon) dir_icon = ui_get_mime_icon("inode/directory", GTK_ICON_SIZE_MENU); gtk_tree_store_append(store_openfiles, &parent, NULL); gtk_tree_store_set(store_openfiles, &parent, DOCUMENTS_ICON, dir_icon, DOCUMENTS_FILENAME, path, DOCUMENTS_SHORTNAME, doc->file_name ? dirname : GEANY_STRING_UNTITLED, -1); g_free(dirname); g_free(path); return &parent; } /* Also sets doc->priv->iter. * This is called recursively in sidebar_openfiles_update_all(). */ void sidebar_openfiles_add(GeanyDocument *doc) { GtkTreeIter *iter = &doc->priv->iter; GtkTreeIter *parent = get_doc_parent(doc); gchar *basename; const GdkColor *color = document_get_status_color(doc); static GdkPixbuf *file_icon = NULL; gtk_tree_store_append(store_openfiles, iter, parent); /* check if new parent */ if (parent && gtk_tree_model_iter_n_children(GTK_TREE_MODEL(store_openfiles), parent) == 1) { GtkTreePath *path; /* expand parent */ path = gtk_tree_model_get_path(GTK_TREE_MODEL(store_openfiles), parent); gtk_tree_view_expand_row(GTK_TREE_VIEW(tv.tree_openfiles), path, TRUE); gtk_tree_path_free(path); } if (!file_icon) file_icon = ui_get_mime_icon("text/plain", GTK_ICON_SIZE_MENU); basename = g_path_get_basename(DOC_FILENAME(doc)); gtk_tree_store_set(store_openfiles, iter, DOCUMENTS_ICON, (doc->file_type && doc->file_type->icon) ? doc->file_type->icon : file_icon, DOCUMENTS_SHORTNAME, basename, DOCUMENTS_DOCUMENT, doc, DOCUMENTS_COLOR, color, DOCUMENTS_FILENAME, DOC_FILENAME(doc), -1); g_free(basename); } static void openfiles_remove(GeanyDocument *doc) { GtkTreeIter *iter = &doc->priv->iter; GtkTreeIter parent; if (gtk_tree_model_iter_parent(GTK_TREE_MODEL(store_openfiles), &parent, iter) && gtk_tree_model_iter_n_children(GTK_TREE_MODEL(store_openfiles), &parent) == 1) gtk_tree_store_remove(store_openfiles, &parent); else gtk_tree_store_remove(store_openfiles, iter); } void sidebar_openfiles_update(GeanyDocument *doc) { GtkTreeIter *iter = &doc->priv->iter; gchar *fname; gtk_tree_model_get(GTK_TREE_MODEL(store_openfiles), iter, DOCUMENTS_FILENAME, &fname, -1); if (utils_str_equal(fname, DOC_FILENAME(doc))) { /* just update color and the icon */ const GdkColor *color = document_get_status_color(doc); GdkPixbuf *icon = doc->file_type->icon; gtk_tree_store_set(store_openfiles, iter, DOCUMENTS_COLOR, color, -1); if (icon) gtk_tree_store_set(store_openfiles, iter, DOCUMENTS_ICON, icon, -1); } else { /* path has changed, so remove and re-add */ GtkTreeSelection *treesel; gboolean sel; treesel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tv.tree_openfiles)); sel = gtk_tree_selection_iter_is_selected(treesel, &doc->priv->iter); openfiles_remove(doc); sidebar_openfiles_add(doc); if (sel) gtk_tree_selection_select_iter(treesel, &doc->priv->iter); } g_free(fname); } void sidebar_openfiles_update_all() { guint i, page_count; GeanyDocument *doc; gtk_tree_store_clear(store_openfiles); page_count = gtk_notebook_get_n_pages(GTK_NOTEBOOK(main_widgets.notebook)); for (i = 0; i < page_count; i++) { doc = document_get_from_page(i); if (G_UNLIKELY(doc == NULL)) continue; sidebar_openfiles_add(doc); } } void sidebar_remove_document(GeanyDocument *doc) { openfiles_remove(doc); if (GTK_IS_WIDGET(doc->priv->tag_tree)) { gtk_widget_destroy(doc->priv->tag_tree); if (GTK_IS_TREE_VIEW(doc->priv->tag_tree)) { /* Because it was ref'd in sidebar_update_tag_list, it needs unref'ing */ g_object_unref((gpointer)doc->priv->tag_tree); } doc->priv->tag_tree = NULL; } } static void on_hide_sidebar(void) { ui_prefs.sidebar_visible = FALSE; ui_sidebar_show_hide(); } static gboolean on_sidebar_display_symbol_list_show(GtkWidget *item) { gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(item), interface_prefs.sidebar_symbol_visible); return FALSE; } static gboolean on_sidebar_display_open_files_show(GtkWidget *item) { gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(item), interface_prefs.sidebar_openfiles_visible); return FALSE; } void sidebar_add_common_menu_items(GtkMenu *menu) { GtkWidget *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 S_ymbol List")); gtk_container_add(GTK_CONTAINER(menu), item); g_signal_connect(item, "expose-event", G_CALLBACK(on_sidebar_display_symbol_list_show), NULL); gtk_widget_show(item); g_signal_connect(item, "activate", G_CALLBACK(on_list_symbol_activate), NULL); item = gtk_check_menu_item_new_with_mnemonic(_("Show _Document List")); gtk_container_add(GTK_CONTAINER(menu), item); g_signal_connect(item, "expose-event", G_CALLBACK(on_sidebar_display_open_files_show), NULL); gtk_widget_show(item); g_signal_connect(item, "activate", G_CALLBACK(on_list_document_activate), NULL); 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(item, "activate", G_CALLBACK(on_hide_sidebar), NULL); } static void on_openfiles_show_paths_activate(GtkCheckMenuItem *item, gpointer user_data) { documents_show_paths = gtk_check_menu_item_get_active(item); sidebar_openfiles_update_all(); } static void on_list_document_activate(GtkCheckMenuItem *item, gpointer user_data) { interface_prefs.sidebar_openfiles_visible = gtk_check_menu_item_get_active(item); ui_sidebar_show_hide(); sidebar_tabs_show_hide(GTK_NOTEBOOK(main_widgets.sidebar_notebook), NULL, 0, NULL); } static void on_list_symbol_activate(GtkCheckMenuItem *item, gpointer user_data) { interface_prefs.sidebar_symbol_visible = gtk_check_menu_item_get_active(item); ui_sidebar_show_hide(); sidebar_tabs_show_hide(GTK_NOTEBOOK(main_widgets.sidebar_notebook), NULL, 0, NULL); } static void on_find_in_files(GtkMenuItem *menuitem, gpointer user_data) { GtkTreeSelection *treesel; GtkTreeIter iter; GtkTreeModel *model; GeanyDocument *doc; gchar *dir; treesel = gtk_tree_view_get_selection(GTK_TREE_VIEW(tv.tree_openfiles)); if (!gtk_tree_selection_get_selected(treesel, &model, &iter)) return; gtk_tree_model_get(model, &iter, DOCUMENTS_DOCUMENT, &doc, -1); if (!doc) { gtk_tree_model_get(model, &iter, DOCUMENTS_FILENAME, &dir, -1); } else dir = g_path_get_dirname(DOC_FILENAME(doc)); search_show_find_in_files_dialog(dir); g_free(dir); } static void create_openfiles_popup_menu(void) { GtkWidget *item; openfiles_popup_menu = gtk_menu_new(); item = gtk_image_menu_item_new_from_stock("gtk-close", NULL); gtk_widget_show(item); gtk_container_add(GTK_CONTAINER(openfiles_popup_menu), item); g_signal_connect(item, "activate", G_CALLBACK(on_openfiles_document_action), GINT_TO_POINTER(OPENFILES_ACTION_REMOVE)); doc_items.close = item; item = gtk_separator_menu_item_new(); gtk_widget_show(item); gtk_container_add(GTK_CONTAINER(openfiles_popup_menu), item); item = gtk_image_menu_item_new_from_stock("gtk-save", NULL); gtk_widget_show(item); gtk_container_add(GTK_CONTAINER(openfiles_popup_menu), item); g_signal_connect(item, "activate", G_CALLBACK(on_openfiles_document_action), GINT_TO_POINTER(OPENFILES_ACTION_SAVE)); doc_items.save = item; item = gtk_image_menu_item_new_with_mnemonic(_("_Reload")); gtk_image_menu_item_set_image(GTK_IMAGE_MENU_ITEM(item), gtk_image_new_from_stock("gtk-revert-to-saved", GTK_ICON_SIZE_MENU)); gtk_widget_show(item); gtk_container_add(GTK_CONTAINER(openfiles_popup_menu), item); g_signal_connect(item, "activate", G_CALLBACK(on_openfiles_document_action), GINT_TO_POINTER(OPENFILES_ACTION_RELOAD)); doc_items.reload = item; item = gtk_separator_menu_item_new(); gtk_widget_show(item); gtk_container_add(GTK_CONTAINER(openfiles_popup_menu), item); item = ui_image_menu_item_new(GTK_STOCK_FIND, _("_Find in Files")); gtk_widget_show(item); gtk_container_add(GTK_CONTAINER(openfiles_popup_menu), item); g_signal_connect(item, "activate", G_CALLBACK(on_find_in_files), NULL); doc_items.find_in_files = item; item = gtk_separator_menu_item_new(); gtk_widget_show(item); gtk_container_add(GTK_CONTAINER(openfiles_popup_menu), item); doc_items.show_paths = gtk_check_menu_item_new_with_mnemonic(_("Show _Paths")); gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(doc_items.show_paths), documents_show_paths); gtk_widget_show(doc_items.show_paths); gtk_container_add(GTK_CONTAINER(openfiles_popup_menu), doc_items.show_paths); g_signal_connect(doc_items.show_paths, "activate", G_CALLBACK(on_openfiles_show_paths_activate), NULL); sidebar_add_common_menu_items(GTK_MENU(openfiles_popup_menu)); } static void unfold_parent(GtkTreeIter *iter) { GtkTreeIter parent; GtkTreePath *path; if (gtk_tree_model_iter_parent(GTK_TREE_MODEL(store_openfiles), &parent, iter)) { path = gtk_tree_model_get_path(GTK_TREE_MODEL(store_openfiles), &parent); gtk_tree_view_expand_row(GTK_TREE_VIEW(tv.tree_openfiles), path, TRUE); gtk_tree_path_free(path); } } /* compares the given data with the doc pointer from the selected row of openfiles * treeview, in case of a match the row is selected and TRUE is returned * (called indirectly from gtk_tree_model_foreach()) */ static gboolean tree_model_find_node(GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, gpointer data) { GeanyDocument *doc; gtk_tree_model_get(GTK_TREE_MODEL(store_openfiles), iter, DOCUMENTS_DOCUMENT, &doc, -1); if (doc == data) { /* unfolding also prevents a strange bug where the selection gets stuck on the parent * when it is collapsed and then switching documents */ unfold_parent(iter); gtk_tree_view_set_cursor(GTK_TREE_VIEW(tv.tree_openfiles), path, NULL, FALSE); return TRUE; } else return FALSE; } void sidebar_select_openfiles_item(GeanyDocument *doc) { gtk_tree_model_foreach(GTK_TREE_MODEL(store_openfiles), tree_model_find_node, doc); } /* callbacks */ static void document_action(GeanyDocument *doc, gint action) { if (! DOC_VALID(doc)) return; switch (action) { case OPENFILES_ACTION_REMOVE: { document_close(doc); break; } case OPENFILES_ACTION_SAVE: { document_save_file(doc, FALSE); break; } case OPENFILES_ACTION_RELOAD: { on_toolbutton_reload_clicked(NULL, NULL); break; } } } static void on_openfiles_document_action(GtkMenuItem *menuitem, gpointer user_data) { GtkTreeIter iter; GtkTreeSelection *selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(tv.tree_openfiles)); GtkTreeModel *model; GeanyDocument *doc; gint action = GPOINTER_TO_INT(user_data); if (gtk_tree_selection_get_selected(selection, &model, &iter)) { gtk_tree_model_get(model, &iter, DOCUMENTS_DOCUMENT, &doc, -1); if (doc) { document_action(doc, action); } else { /* parent item selected */ GtkTreeIter child; gint i = gtk_tree_model_iter_n_children(model, &iter) - 1; while (i >= 0 && gtk_tree_model_iter_nth_child(model, &child, &iter, i)) { gtk_tree_model_get(model, &child, DOCUMENTS_DOCUMENT, &doc, -1); document_action(doc, action); i--; } } } } static void change_focus_to_editor(GeanyDocument *doc, GtkWidget *source_widget) { if (may_steal_focus) document_try_focus(doc, source_widget); may_steal_focus = FALSE; } static gboolean openfiles_go_to_selection(GtkTreeSelection *selection, guint keyval) { GtkTreeIter iter; GtkTreeModel *model; GeanyDocument *doc = NULL; /* use switch_notebook_page to ignore changing the notebook page because it is already done */ if (gtk_tree_selection_get_selected(selection, &model, &iter) && ! ignore_callback) { gtk_tree_model_get(model, &iter, DOCUMENTS_DOCUMENT, &doc, -1); if (! doc) return FALSE; /* parent */ /* switch to the doc and grab the focus */ document_show_tab(doc); if (keyval != GDK_space) change_focus_to_editor(doc, tv.tree_openfiles); } return FALSE; } static gboolean taglist_go_to_selection(GtkTreeSelection *selection, guint keyval) { GtkTreeIter iter; GtkTreeModel *model; gint line = 0; if (gtk_tree_selection_get_selected(selection, &model, &iter)) { TMTag *tag; gtk_tree_model_get(model, &iter, SYMBOLS_COLUMN_TAG, &tag, -1); if (! tag) return FALSE; line = tag->atts.entry.line; if (line > 0) { GeanyDocument *doc = document_get_current(); if (doc != NULL) { navqueue_goto_line(doc, doc, line); if (keyval != GDK_space) change_focus_to_editor(doc, NULL); } } tm_tag_unref(tag); } return FALSE; } static gboolean sidebar_key_press_cb(GtkWidget *widget, GdkEventKey *event, gpointer user_data) { may_steal_focus = FALSE; if (ui_is_keyval_enter_or_return(event->keyval) || event->keyval == GDK_space) { GtkWidgetClass *widget_class = GTK_WIDGET_GET_CLASS(widget); GtkTreeSelection *selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(widget)); may_steal_focus = TRUE; /* force the TreeView handler to run before us for it to do its job (selection & stuff). * doing so will prevent further handlers to be run in most cases, but the only one is our * own, so guess it's fine. */ if (widget_class->key_press_event) widget_class->key_press_event(widget, event); if (widget == tv.tree_openfiles) /* tag and doc list have separate handlers */ openfiles_go_to_selection(selection, event->keyval); else taglist_go_to_selection(selection, event->keyval); return TRUE; } return FALSE; } static gboolean sidebar_button_press_cb(GtkWidget *widget, GdkEventButton *event, G_GNUC_UNUSED gpointer user_data) { GtkTreeSelection *selection; GtkWidgetClass *widget_class = GTK_WIDGET_GET_CLASS(widget); gboolean handled = FALSE; /* force the TreeView handler to run before us for it to do its job (selection & stuff). * doing so will prevent further handlers to be run in most cases, but the only one is our own, * so guess it's fine. */ if (widget_class->button_press_event) handled = widget_class->button_press_event(widget, event); selection = gtk_tree_view_get_selection(GTK_TREE_VIEW(widget)); may_steal_focus = TRUE; if (event->type == GDK_2BUTTON_PRESS) { /* double click on parent node(section) expands/collapses it */ GtkTreeModel *model; GtkTreeIter iter; if (gtk_tree_selection_get_selected(selection, &model, &iter)) { if (gtk_tree_model_iter_has_child(model, &iter)) { GtkTreePath *path = gtk_tree_model_get_path(model, &iter); if (gtk_tree_view_row_expanded(GTK_TREE_VIEW(widget), path)) gtk_tree_view_collapse_row(GTK_TREE_VIEW(widget), path); else gtk_tree_view_expand_row(GTK_TREE_VIEW(widget), path, FALSE); gtk_tree_path_free(path); return TRUE; } } } else if (event->button == 1) { /* allow reclicking of taglist treeview item */ if (widget == tv.tree_openfiles) openfiles_go_to_selection(selection, 0); else taglist_go_to_selection(selection, 0); handled = TRUE; } else if (event->button == 3) { if (widget == tv.tree_openfiles) { if (!openfiles_popup_menu) create_openfiles_popup_menu(); /* update menu item sensitivity */ documents_menu_update(selection); gtk_menu_popup(GTK_MENU(openfiles_popup_menu), NULL, NULL, NULL, NULL, event->button, event->time); } else { gtk_menu_popup(GTK_MENU(tv.popup_taglist), NULL, NULL, NULL, NULL, event->button, event->time); } handled = TRUE; } return handled; } static void documents_menu_update(GtkTreeSelection *selection) { GtkTreeModel *model; GtkTreeIter iter; gboolean sel, path; gchar *shortname = NULL; GeanyDocument *doc = NULL; /* maybe no selection e.g. if ctrl-click deselected */ sel = gtk_tree_selection_get_selected(selection, &model, &iter); if (sel) { gtk_tree_model_get(model, &iter, DOCUMENTS_DOCUMENT, &doc, DOCUMENTS_SHORTNAME, &shortname, -1); } path = NZV(shortname) && (g_path_is_absolute(shortname) || (app->project && g_str_has_prefix(shortname, app->project->name))); /* can close all, save all (except shortname), but only reload individually ATM */ gtk_widget_set_sensitive(doc_items.close, sel); gtk_widget_set_sensitive(doc_items.save, (doc && doc->real_path) || path); gtk_widget_set_sensitive(doc_items.reload, doc && doc->real_path); gtk_widget_set_sensitive(doc_items.find_in_files, sel); g_free(shortname); gtk_check_menu_item_set_active(GTK_CHECK_MENU_ITEM(doc_items.show_paths), documents_show_paths); } static StashGroup *stash_group = NULL; static void on_load_settings(void) { tag_window = ui_lookup_widget(main_widgets.window, "scrolledwindow2"); prepare_openfiles(); /* note: ui_prefs.sidebar_page is reapplied after plugins are loaded */ stash_group_display(stash_group, NULL); sidebar_tabs_show_hide(GTK_NOTEBOOK(main_widgets.sidebar_notebook), NULL, 0, NULL); } static void on_save_settings(void) { stash_group_update(stash_group, NULL); sidebar_tabs_show_hide(GTK_NOTEBOOK(main_widgets.sidebar_notebook), NULL, 0, NULL); } void sidebar_init(void) { StashGroup *group; group = stash_group_new(PACKAGE); stash_group_add_boolean(group, &documents_show_paths, "documents_show_paths", TRUE); stash_group_add_widget_property(group, &ui_prefs.sidebar_page, "sidebar_page", GINT_TO_POINTER(0), main_widgets.sidebar_notebook, "page", 0); configuration_add_pref_group(group, FALSE); stash_group = group; /* delay building documents treeview until sidebar font has been read */ g_signal_connect(geany_object, "load-settings", on_load_settings, NULL); g_signal_connect(geany_object, "save-settings", on_save_settings, NULL); g_signal_connect(main_widgets.sidebar_notebook, "page-added", G_CALLBACK(sidebar_tabs_show_hide), NULL); g_signal_connect(main_widgets.sidebar_notebook, "page-removed", G_CALLBACK(sidebar_tabs_show_hide), NULL); /* tabs may have changed when sidebar is reshown */ g_signal_connect(main_widgets.sidebar_notebook, "show", G_CALLBACK(sidebar_tabs_show_hide), NULL); sidebar_tabs_show_hide(GTK_NOTEBOOK(main_widgets.sidebar_notebook), NULL, 0, NULL); } #define WIDGET(w) w && GTK_IS_WIDGET(w) void sidebar_finalize(void) { if (WIDGET(tv.default_tag_tree)) { g_object_unref(tv.default_tag_tree); /* This is not exactly clean, default_tag_tree's ref_count is 2 when it is shown, * 1 oherwise. We should probably handle the ref_count more accurate. */ if (WIDGET(tv.default_tag_tree)) gtk_widget_destroy(tv.default_tag_tree); } if (WIDGET(tv.popup_taglist)) gtk_widget_destroy(tv.popup_taglist); if (WIDGET(openfiles_popup_menu)) gtk_widget_destroy(openfiles_popup_menu); } void sidebar_focus_openfiles_tab(void) { if (ui_prefs.sidebar_visible && interface_prefs.sidebar_openfiles_visible) { GtkNotebook *notebook = GTK_NOTEBOOK(main_widgets.sidebar_notebook); gtk_notebook_set_current_page(notebook, TREEVIEW_OPENFILES); gtk_widget_grab_focus(tv.tree_openfiles); } } void sidebar_focus_symbols_tab(void) { if (ui_prefs.sidebar_visible && interface_prefs.sidebar_symbol_visible) { GtkNotebook *notebook = GTK_NOTEBOOK(main_widgets.sidebar_notebook); GtkWidget *symbol_list_scrollwin = gtk_notebook_get_nth_page(notebook, TREEVIEW_SYMBOL); gtk_notebook_set_current_page(notebook, TREEVIEW_SYMBOL); gtk_widget_grab_focus(gtk_bin_get_child(GTK_BIN(symbol_list_scrollwin))); } } static void sidebar_tabs_show_hide(GtkNotebook *notebook, GtkWidget *child, guint page_num, gpointer data) { gint tabs = gtk_notebook_get_n_pages(notebook); if (interface_prefs.sidebar_symbol_visible == FALSE) tabs--; if (interface_prefs.sidebar_openfiles_visible == FALSE) tabs--; gtk_notebook_set_show_tabs(notebook, (tabs > 1)); }