/* -*- Mode: C; tab-width: 8; indent-tabs-mode: nil; c-basic-offset: 8 -*- */ /* * Copyright (C) 2001-2003 CodeFactory AB * Copyright (C) 2001-2003 Mikael Hallendal * Copyright (C) 2005-2008 Imendio AB * Copyright (C) 2010 Lanedo GmbH * Copyright (C) 2013 Aleksander Morgado * Copyright (C) 2015, 2017, 2018 Sébastien Wilmet * * 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, see . */ #include "dh-sidebar.h" #include "dh-book.h" #include "dh-book-tree.h" #include "dh-keyword-model.h" /** * SECTION:dh-sidebar * @Title: DhSidebar * @Short_description: The sidebar * * In the Devhelp application, there is one #DhSidebar per main window, * displayed in the left side panel. * * A #DhSidebar contains: * - a #GtkSearchEntry at the top; * - a #DhBookTree (a subclass of #GtkTreeView); * - another #GtkTreeView (displaying a list, not a tree) with a #DhKeywordModel * as its model. * * When the #GtkSearchEntry is empty, the #DhBookTree is shown. When the * #GtkSearchEntry is not empty, it shows the search results in the other * #GtkTreeView. The two #GtkTreeView's cannot be both visible at the same time, * it's either one or the other. * * The #DhSidebar::link-selected signal is emitted when one element in one of * the #GtkTreeView's is selected. When that happens, the Devhelp application * opens the link in a #WebKitWebView shown at the right side of the main * window. */ typedef struct { /* A GtkSearchEntry. */ GtkEntry *entry; DhBookTree *book_tree; GtkScrolledWindow *sw_book_tree; DhKeywordModel *hitlist_model; GtkTreeView *hitlist_view; GtkScrolledWindow *sw_hitlist; guint idle_complete_id; guint idle_search_id; } DhSidebarPrivate; enum { SIGNAL_LINK_SELECTED, N_SIGNALS }; static guint signals[N_SIGNALS] = { 0 }; G_DEFINE_TYPE_WITH_PRIVATE (DhSidebar, dh_sidebar, GTK_TYPE_GRID) static void dh_sidebar_dispose (GObject *object) { DhSidebarPrivate *priv = dh_sidebar_get_instance_private (DH_SIDEBAR (object)); g_clear_object (&priv->hitlist_model); if (priv->idle_complete_id != 0) { g_source_remove (priv->idle_complete_id); priv->idle_complete_id = 0; } if (priv->idle_search_id != 0) { g_source_remove (priv->idle_search_id); priv->idle_search_id = 0; } G_OBJECT_CLASS (dh_sidebar_parent_class)->dispose (object); } static void dh_sidebar_class_init (DhSidebarClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); object_class->dispose = dh_sidebar_dispose; /** * DhSidebar::link-selected: * @sidebar: a #DhSidebar. * @link: the selected #DhLink. */ signals[SIGNAL_LINK_SELECTED] = g_signal_new ("link-selected", G_TYPE_FROM_CLASS (klass), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (DhSidebarClass, link_selected), NULL, NULL, NULL, G_TYPE_NONE, 1, DH_TYPE_LINK); } /******************************************************************************/ static gboolean search_idle_cb (gpointer user_data) { DhSidebar *sidebar = DH_SIDEBAR (user_data); DhSidebarPrivate *priv = dh_sidebar_get_instance_private (sidebar); const gchar *search_text; const gchar *book_id; DhLink *book_link; DhLink *exact_link; priv->idle_search_id = 0; search_text = gtk_entry_get_text (priv->entry); book_link = dh_book_tree_get_selected_book (priv->book_tree); book_id = book_link != NULL ? dh_link_get_book_id (book_link) : NULL; /* Disconnect the model, see the doc of dh_keyword_model_filter(). */ gtk_tree_view_set_model (priv->hitlist_view, NULL); exact_link = dh_keyword_model_filter (priv->hitlist_model, search_text, book_id, NULL); gtk_tree_view_set_model (priv->hitlist_view, GTK_TREE_MODEL (priv->hitlist_model)); if (exact_link != NULL) g_signal_emit (sidebar, signals[SIGNAL_LINK_SELECTED], 0, exact_link); if (book_link != NULL) dh_link_unref (book_link); return G_SOURCE_REMOVE; } static void setup_search_idle (DhSidebar *sidebar) { DhSidebarPrivate *priv = dh_sidebar_get_instance_private (sidebar); if (priv->idle_search_id == 0) priv->idle_search_id = g_idle_add (search_idle_cb, sidebar); } static void book_manager_changed_cb (DhSidebar *sidebar) { /* Update current search if any. */ setup_search_idle (sidebar); } /******************************************************************************/ static void hitlist_selection_changed_cb (GtkTreeSelection *selection, DhSidebar *sidebar) { DhSidebarPrivate *priv = dh_sidebar_get_instance_private (sidebar); GtkTreeIter iter; if (gtk_tree_selection_get_selected (selection, NULL, &iter)) { DhLink *link; gtk_tree_model_get (GTK_TREE_MODEL (priv->hitlist_model), &iter, DH_KEYWORD_MODEL_COL_LINK, &link, -1); g_signal_emit (sidebar, signals[SIGNAL_LINK_SELECTED], 0, link); dh_link_unref (link); } } /* Make it possible to jump back to the currently selected item, useful when the * html view has been scrolled away. */ static gboolean hitlist_button_press_cb (GtkTreeView *hitlist_view, GdkEventButton *event, DhSidebar *sidebar) { DhSidebarPrivate *priv = dh_sidebar_get_instance_private (sidebar); GtkTreePath *path; GtkTreeIter iter; DhLink *link; gtk_tree_view_get_path_at_pos (hitlist_view, event->x, event->y, &path, NULL, NULL, NULL); if (path == NULL) return GDK_EVENT_PROPAGATE; gtk_tree_model_get_iter (GTK_TREE_MODEL (priv->hitlist_model), &iter, path); gtk_tree_path_free (path); gtk_tree_model_get (GTK_TREE_MODEL (priv->hitlist_model), &iter, DH_KEYWORD_MODEL_COL_LINK, &link, -1); g_signal_emit (sidebar, signals[SIGNAL_LINK_SELECTED], 0, link); dh_link_unref (link); /* Always propagate the event so the tree view can update * the selection etc. */ return GDK_EVENT_PROPAGATE; } static gboolean entry_key_press_event_cb (GtkEntry *entry, GdkEventKey *event, DhSidebar *sidebar) { DhSidebarPrivate *priv = dh_sidebar_get_instance_private (sidebar); if (event->keyval == GDK_KEY_Tab) { if (event->state & GDK_CONTROL_MASK) { if (gtk_widget_is_visible (GTK_WIDGET (priv->hitlist_view))) gtk_widget_grab_focus (GTK_WIDGET (priv->hitlist_view)); } else { gtk_editable_select_region (GTK_EDITABLE (entry), 0, 0); gtk_editable_set_position (GTK_EDITABLE (entry), -1); } return GDK_EVENT_STOP; } if (event->keyval == GDK_KEY_Return || event->keyval == GDK_KEY_KP_Enter) { GtkTreeIter iter; DhLink *link; gchar *name; /* Get the first entry found. * * FIXME: is it really useful to do that? If there is an exact * match it already gets selected, so it seems that the feature * here just selects a random symbol (the one that appears to be * the first in the list). * I've never used this feature -- swilmet. * This has been implemented in * commit 455440a93d1b55d5a1e53ecabb2ee33093eec965 * and https://bugzilla.gnome.org/show_bug.cgi?id=114558 * but maybe at that time the search didn't jump to the exact * match if there was one. */ if (gtk_widget_is_visible (GTK_WIDGET (priv->hitlist_view)) && gtk_tree_model_get_iter_first (GTK_TREE_MODEL (priv->hitlist_model), &iter)) { gtk_tree_model_get (GTK_TREE_MODEL (priv->hitlist_model), &iter, DH_KEYWORD_MODEL_COL_LINK, &link, DH_KEYWORD_MODEL_COL_NAME, &name, -1); gtk_entry_set_text (entry, name); g_free (name); gtk_editable_select_region (GTK_EDITABLE (entry), 0, 0); gtk_editable_set_position (GTK_EDITABLE (entry), -1); g_signal_emit (sidebar, signals[SIGNAL_LINK_SELECTED], 0, link); dh_link_unref (link); return GDK_EVENT_STOP; } } return GDK_EVENT_PROPAGATE; } static void entry_changed_cb (GtkEntry *entry, DhSidebar *sidebar) { DhSidebarPrivate *priv = dh_sidebar_get_instance_private (sidebar); const gchar *search_text; search_text = gtk_entry_get_text (entry); /* We don't want a delay when the search text becomes empty, to show the * book tree. So do it here and not in entry_search_changed_cb(). */ if (search_text == NULL || search_text[0] == '\0') { gtk_widget_hide (GTK_WIDGET (priv->sw_hitlist)); gtk_widget_show (GTK_WIDGET (priv->sw_book_tree)); } } static void entry_search_changed_cb (GtkSearchEntry *search_entry, DhSidebar *sidebar) { DhSidebarPrivate *priv = dh_sidebar_get_instance_private (sidebar); const gchar *search_text; search_text = gtk_entry_get_text (GTK_ENTRY (search_entry)); if (search_text != NULL && search_text[0] != '\0') { gtk_widget_hide (GTK_WIDGET (priv->sw_book_tree)); gtk_widget_show (GTK_WIDGET (priv->sw_hitlist)); setup_search_idle (sidebar); } } static gboolean complete_idle_cb (gpointer user_data) { DhSidebar *sidebar = DH_SIDEBAR (user_data); DhSidebarPrivate *priv = dh_sidebar_get_instance_private (sidebar); DhBookManager *book_manager; GList *books; GList *l; GList *completion_objects = NULL; const gchar *search_text; gchar *completed; book_manager = dh_book_manager_get_singleton (); books = dh_book_manager_get_books (book_manager); for (l = books; l != NULL; l = l->next) { DhBook *cur_book = DH_BOOK (l->data); if (dh_book_get_enabled (cur_book)) { DhCompletion *completion; completion = dh_book_get_completion (cur_book); completion_objects = g_list_prepend (completion_objects, completion); } } search_text = gtk_entry_get_text (priv->entry); completed = dh_completion_aggregate_complete (completion_objects, search_text); if (completed != NULL) { guint16 n_chars_before; n_chars_before = gtk_entry_get_text_length (priv->entry); gtk_entry_set_text (priv->entry, completed); gtk_editable_set_position (GTK_EDITABLE (priv->entry), n_chars_before); gtk_editable_select_region (GTK_EDITABLE (priv->entry), n_chars_before, -1); } g_list_free (completion_objects); g_free (completed); priv->idle_complete_id = 0; return G_SOURCE_REMOVE; } static void entry_insert_text_cb (GtkEntry *entry, const gchar *text, gint length, gint *position, DhSidebar *sidebar) { DhSidebarPrivate *priv = dh_sidebar_get_instance_private (sidebar); if (priv->idle_complete_id == 0) priv->idle_complete_id = g_idle_add (complete_idle_cb, sidebar); } static void entry_stop_search_cb (GtkSearchEntry *entry, gpointer user_data) { gtk_entry_set_text (GTK_ENTRY (entry), ""); } static void hitlist_cell_data_func (GtkTreeViewColumn *tree_column, GtkCellRenderer *cell, GtkTreeModel *hitlist_model, GtkTreeIter *iter, gpointer data) { DhLink *link; DhLinkType link_type; PangoStyle style; PangoWeight weight; gboolean current_book_flag; gchar *name; gtk_tree_model_get (hitlist_model, iter, DH_KEYWORD_MODEL_COL_LINK, &link, DH_KEYWORD_MODEL_COL_CURRENT_BOOK_FLAG, ¤t_book_flag, -1); if (dh_link_get_flags (link) & DH_LINK_FLAGS_DEPRECATED) style = PANGO_STYLE_ITALIC; else style = PANGO_STYLE_NORMAL; /* Matches on the current book are given in bold. Note that we check the * current book as it was given to the DhKeywordModel. Do *not* rely on * the current book as given by the DhSidebar, as that will change * whenever a hit is clicked. */ if (current_book_flag) weight = PANGO_WEIGHT_BOLD; else weight = PANGO_WEIGHT_NORMAL; link_type = dh_link_get_link_type (link); if (link_type == DH_LINK_TYPE_STRUCT || link_type == DH_LINK_TYPE_PROPERTY || link_type == DH_LINK_TYPE_SIGNAL) { name = g_markup_printf_escaped ("%s (%s)", dh_link_get_name (link), dh_link_type_to_string (link_type)); } else { name = g_markup_printf_escaped ("%s", dh_link_get_name (link)); } g_object_set (cell, "markup", name, "style", style, "weight", weight, NULL); dh_link_unref (link); g_free (name); } static void book_tree_link_selected_cb (DhBookTree *book_tree, DhLink *link, DhSidebar *sidebar) { g_signal_emit (sidebar, signals[SIGNAL_LINK_SELECTED], 0, link); } static void dh_sidebar_init (DhSidebar *sidebar) { DhSidebarPrivate *priv = dh_sidebar_get_instance_private (sidebar); GtkCellRenderer *cell; DhBookManager *book_manager; gtk_orientable_set_orientation (GTK_ORIENTABLE (sidebar), GTK_ORIENTATION_VERTICAL); gtk_widget_set_hexpand (GTK_WIDGET (sidebar), TRUE); gtk_widget_set_vexpand (GTK_WIDGET (sidebar), TRUE); /* Setup the search entry */ priv->entry = GTK_ENTRY (gtk_search_entry_new ()); gtk_widget_set_hexpand (GTK_WIDGET (priv->entry), TRUE); g_object_set (priv->entry, "margin", 6, NULL); gtk_container_add (GTK_CONTAINER (sidebar), GTK_WIDGET (priv->entry)); g_signal_connect (priv->entry, "key-press-event", G_CALLBACK (entry_key_press_event_cb), sidebar); g_signal_connect (priv->entry, "changed", G_CALLBACK (entry_changed_cb), sidebar); g_signal_connect (priv->entry, "search-changed", G_CALLBACK (entry_search_changed_cb), sidebar); g_signal_connect (priv->entry, "insert-text", G_CALLBACK (entry_insert_text_cb), sidebar); g_signal_connect (priv->entry, "stop-search", G_CALLBACK (entry_stop_search_cb), NULL); /* Setup hitlist */ priv->hitlist_model = dh_keyword_model_new (); priv->hitlist_view = GTK_TREE_VIEW (gtk_tree_view_new ()); gtk_tree_view_set_model (priv->hitlist_view, GTK_TREE_MODEL (priv->hitlist_model)); gtk_tree_view_set_headers_visible (priv->hitlist_view, FALSE); gtk_tree_view_set_enable_search (priv->hitlist_view, FALSE); gtk_widget_show (GTK_WIDGET (priv->hitlist_view)); g_signal_connect (priv->hitlist_view, "button-press-event", G_CALLBACK (hitlist_button_press_cb), sidebar); g_signal_connect (gtk_tree_view_get_selection (priv->hitlist_view), "changed", G_CALLBACK (hitlist_selection_changed_cb), sidebar); cell = gtk_cell_renderer_text_new (); g_object_set (cell, "ellipsize", PANGO_ELLIPSIZE_END, NULL); gtk_tree_view_insert_column_with_data_func (priv->hitlist_view, -1, NULL, cell, hitlist_cell_data_func, sidebar, NULL); /* Hitlist packing */ priv->sw_hitlist = GTK_SCROLLED_WINDOW (gtk_scrolled_window_new (NULL, NULL)); gtk_widget_set_no_show_all (GTK_WIDGET (priv->sw_hitlist), TRUE); gtk_scrolled_window_set_policy (priv->sw_hitlist, GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC); gtk_container_add (GTK_CONTAINER (priv->sw_hitlist), GTK_WIDGET (priv->hitlist_view)); gtk_widget_set_hexpand (GTK_WIDGET (priv->sw_hitlist), TRUE); gtk_widget_set_vexpand (GTK_WIDGET (priv->sw_hitlist), TRUE); gtk_container_add (GTK_CONTAINER (sidebar), GTK_WIDGET (priv->sw_hitlist)); /* Setup book manager */ book_manager = dh_book_manager_get_singleton (); g_signal_connect_object (book_manager, "book-created", G_CALLBACK (book_manager_changed_cb), sidebar, G_CONNECT_SWAPPED); g_signal_connect_object (book_manager, "book-enabled", G_CALLBACK (book_manager_changed_cb), sidebar, G_CONNECT_SWAPPED); g_signal_connect_object (book_manager, "book-deleted", G_CALLBACK (book_manager_changed_cb), sidebar, G_CONNECT_SWAPPED); g_signal_connect_object (book_manager, "book-disabled", G_CALLBACK (book_manager_changed_cb), sidebar, G_CONNECT_SWAPPED); /* Setup the book tree */ priv->sw_book_tree = GTK_SCROLLED_WINDOW (gtk_scrolled_window_new (NULL, NULL)); gtk_widget_show (GTK_WIDGET (priv->sw_book_tree)); gtk_widget_set_no_show_all (GTK_WIDGET (priv->sw_book_tree), TRUE); gtk_scrolled_window_set_policy (priv->sw_book_tree, GTK_POLICY_NEVER, GTK_POLICY_AUTOMATIC); priv->book_tree = dh_book_tree_new (); gtk_widget_show (GTK_WIDGET (priv->book_tree)); g_signal_connect (priv->book_tree, "link-selected", G_CALLBACK (book_tree_link_selected_cb), sidebar); gtk_container_add (GTK_CONTAINER (priv->sw_book_tree), GTK_WIDGET (priv->book_tree)); gtk_widget_set_hexpand (GTK_WIDGET (priv->sw_book_tree), TRUE); gtk_widget_set_vexpand (GTK_WIDGET (priv->sw_book_tree), TRUE); gtk_container_add (GTK_CONTAINER (sidebar), GTK_WIDGET (priv->sw_book_tree)); gtk_widget_show_all (GTK_WIDGET (sidebar)); } /** * dh_sidebar_new: * @book_manager: (nullable): a #DhBookManager. This parameter is deprecated, * you should just pass %NULL. * * Returns: (transfer floating): a new #DhSidebar widget. */ GtkWidget * dh_sidebar_new (DhBookManager *book_manager) { return g_object_new (DH_TYPE_SIDEBAR, NULL); } /** * dh_sidebar_select_uri: * @sidebar: a #DhSidebar. * @uri: the URI to select. */ void dh_sidebar_select_uri (DhSidebar *sidebar, const gchar *uri) { DhSidebarPrivate *priv; g_return_if_fail (DH_IS_SIDEBAR (sidebar)); g_return_if_fail (uri != NULL); priv = dh_sidebar_get_instance_private (sidebar); dh_book_tree_select_uri (priv->book_tree, uri); } /** * dh_sidebar_set_search_string: * @sidebar: a #DhSidebar. * @str: the string to search. */ void dh_sidebar_set_search_string (DhSidebar *sidebar, const gchar *str) { DhSidebarPrivate *priv; g_return_if_fail (DH_IS_SIDEBAR (sidebar)); g_return_if_fail (str != NULL); priv = dh_sidebar_get_instance_private (sidebar); gtk_entry_set_text (priv->entry, str); gtk_editable_select_region (GTK_EDITABLE (priv->entry), 0, 0); gtk_editable_set_position (GTK_EDITABLE (priv->entry), -1); /* If the GtkEntry text was already equal to @str, the * GtkEditable::changed signal was not emitted, so force to emit it to * call entry_changed_cb() and entry_search_changed_cb(), forcing a new * search. If an exact match is found, the DhSidebar::link-selected * signal will be emitted, to re-jump to that symbol (even if the * GtkEntry text was equal, it doesn't mean that the WebKitWebView was * showing the exact match). * https://bugzilla.gnome.org/show_bug.cgi?id=776596 */ g_signal_emit_by_name (priv->entry, "changed"); } /** * dh_sidebar_set_search_focus: * @sidebar: a #DhSidebar. * * Gives the focus to the search entry. */ void dh_sidebar_set_search_focus (DhSidebar *sidebar) { DhSidebarPrivate *priv; g_return_if_fail (DH_IS_SIDEBAR (sidebar)); priv = dh_sidebar_get_instance_private (sidebar); gtk_widget_grab_focus (GTK_WIDGET (priv->entry)); }