/* * Copyright (C) 2002 Jorn Baayen * * 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., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. * * $Id$ */ #include #include #include #include #include #include #include "eggtreemodelfilter.h" #include "ephy-node-view.h" #include "ephy-tree-model-sort.h" #include "eggtreemultidnd.h" #include "ephy-dnd.h" #include "ephy-marshal.h" #include "string.h" static void ephy_node_view_class_init (EphyNodeViewClass *klass); static void ephy_node_view_init (EphyNodeView *view); static void ephy_node_view_finalize (GObject *object); static void ephy_node_view_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec); static void ephy_node_view_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec); struct EphyNodeViewPrivate { EphyNode *root; EphyTreeModelNode *nodemodel; GtkTreeModel *filtermodel; GtkTreeModel *sortmodel; GtkCellRenderer *editable_renderer; GtkTreeViewColumn *editable_column; EphyTreeModelNodeColumn editable_node_column; EphyNodeFilter *filter; GtkTargetList *drag_targets; int default_sort_column_id; int priority_prop_id; gboolean editing; int editable_property; int searchable_data_column; }; enum { NODE_ACTIVATED, NODE_SELECTED, NODE_DROPPED, SHOW_POPUP, LAST_SIGNAL }; enum { PROP_0, PROP_ROOT, PROP_FILTER }; static GObjectClass *parent_class = NULL; static guint ephy_node_view_signals[LAST_SIGNAL] = { 0 }; GType ephy_node_view_get_type (void) { static GType ephy_node_view_type = 0; if (ephy_node_view_type == 0) { static const GTypeInfo our_info = { sizeof (EphyNodeViewClass), NULL, NULL, (GClassInitFunc) ephy_node_view_class_init, NULL, NULL, sizeof (EphyNodeView), 0, (GInstanceInitFunc) ephy_node_view_init }; ephy_node_view_type = g_type_register_static (GTK_TYPE_TREE_VIEW, "EphyNodeView", &our_info, 0); } return ephy_node_view_type; } static void ephy_node_view_class_init (EphyNodeViewClass *klass) { GObjectClass *object_class = G_OBJECT_CLASS (klass); parent_class = g_type_class_peek_parent (klass); object_class->finalize = ephy_node_view_finalize; object_class->set_property = ephy_node_view_set_property; object_class->get_property = ephy_node_view_get_property; g_object_class_install_property (object_class, PROP_ROOT, g_param_spec_pointer ("root", "Root node", "Root node", G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); g_object_class_install_property (object_class, PROP_FILTER, g_param_spec_object ("filter", "Filter object", "Filter object", EPHY_TYPE_NODE_FILTER, G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY)); ephy_node_view_signals[NODE_ACTIVATED] = g_signal_new ("node_activated", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (EphyNodeViewClass, node_activated), NULL, NULL, g_cclosure_marshal_VOID__POINTER, G_TYPE_NONE, 1, G_TYPE_POINTER); ephy_node_view_signals[NODE_SELECTED] = g_signal_new ("node_selected", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (EphyNodeViewClass, node_selected), NULL, NULL, g_cclosure_marshal_VOID__POINTER, G_TYPE_NONE, 1, G_TYPE_POINTER); ephy_node_view_signals[NODE_DROPPED] = g_signal_new ("node_dropped", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (EphyNodeViewClass, node_dropped), NULL, NULL, ephy_marshal_VOID__POINTER_POINTER, G_TYPE_NONE, 2, G_TYPE_POINTER, G_TYPE_POINTER); ephy_node_view_signals[SHOW_POPUP] = g_signal_new ("show_popup", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (EphyNodeViewClass, show_popup), NULL, NULL, g_cclosure_marshal_VOID__VOID, G_TYPE_NONE, 0); } static void ephy_node_view_finalize (GObject *object) { EphyNodeView *view; g_return_if_fail (object != NULL); g_return_if_fail (EPHY_IS_NODE_VIEW (object)); view = EPHY_NODE_VIEW (object); g_return_if_fail (view->priv != NULL); g_object_unref (G_OBJECT (view->priv->sortmodel)); g_object_unref (G_OBJECT (view->priv->filtermodel)); g_object_unref (G_OBJECT (view->priv->nodemodel)); g_free (view->priv); G_OBJECT_CLASS (parent_class)->finalize (object); } static EphyNode * get_node_from_path (EphyNodeView *view, GtkTreePath *path) { EphyNode *node; GtkTreeIter iter, iter2; gtk_tree_model_get_iter (view->priv->sortmodel, &iter, path); gtk_tree_model_sort_convert_iter_to_child_iter (GTK_TREE_MODEL_SORT (view->priv->sortmodel), &iter2, &iter); egg_tree_model_filter_convert_iter_to_child_iter (EGG_TREE_MODEL_FILTER (view->priv->filtermodel), &iter, &iter2); node = ephy_tree_model_node_node_from_iter (view->priv->nodemodel, &iter); return node; } static gboolean drag_motion_cb (GtkWidget *widget, GdkDragContext *context, gint x, gint y, guint time, EphyNodeView *view) { EphyNode *node; GtkTreePath *path = NULL; GtkTreeViewDropPosition pos; gboolean res; EphyNodeViewPriority priority; g_signal_stop_emission_by_name (widget, "drag_motion"); res = gtk_tree_view_get_dest_row_at_pos (GTK_TREE_VIEW (widget), x, y, &path, &pos); if (!res) return TRUE; node = get_node_from_path (view, path); priority = ephy_node_get_property_int (node, view->priv->priority_prop_id); if (priority != EPHY_NODE_VIEW_ALL_PRIORITY && priority != EPHY_NODE_VIEW_SPECIAL_PRIORITY) { gtk_tree_view_set_drag_dest_row (GTK_TREE_VIEW (widget), path, GTK_TREE_VIEW_DROP_INTO_OR_AFTER); gdk_drag_status (context, context->suggested_action, time); } else { gdk_drag_status (context, 0, time); } gtk_tree_path_free (path); return TRUE; } static gboolean drag_drop_cb (GtkWidget *widget, GdkDragContext *context, gint x, gint y, guint time, EphyNodeView *view) { GdkAtom target; g_signal_stop_emission_by_name (widget, "drag_drop"); target = gtk_drag_dest_find_target (widget, context, view->priv->drag_targets); if (target != GDK_NONE) { gtk_drag_get_data (widget, context, target, time); } return TRUE; } static gboolean drag_data_received_cb (GtkWidget *widget, GdkDragContext *context, gint x, gint y, GtkSelectionData *selection_data, guint info, guint time, EphyNodeView *view) { GtkTreePath *path = NULL; GtkTreeViewDropPosition pos; g_signal_stop_emission_by_name (widget, "drag_data_received"); if (gtk_tree_view_get_dest_row_at_pos (GTK_TREE_VIEW (widget), x, y, &path, &pos)) { EphyNode *node; GList *src_nodes; node = get_node_from_path (view, path); src_nodes = ephy_dnd_node_list_extract_nodes (selection_data->data); g_signal_emit (G_OBJECT (view), ephy_node_view_signals[NODE_DROPPED], 0, node, src_nodes); g_list_free (src_nodes); gtk_tree_path_free (path); } return TRUE; } void ephy_node_view_enable_drag_dest (EphyNodeView *view, GtkTargetEntry *types, int n_types) { GtkWidget *treeview; g_return_if_fail (view != NULL); treeview = GTK_WIDGET (view); gtk_tree_view_enable_model_drag_dest (GTK_TREE_VIEW (treeview), types, n_types, GDK_ACTION_COPY); view->priv->drag_targets = gtk_target_list_new (types, n_types); g_signal_connect (treeview, "drag_data_received", G_CALLBACK (drag_data_received_cb), view); g_signal_connect (treeview, "drag_drop", G_CALLBACK (drag_drop_cb), view); g_signal_connect (treeview, "drag_motion", G_CALLBACK (drag_motion_cb), view); } static void filter_changed_cb (EphyNodeFilter *filter, EphyNodeView *view) { GtkWidget *window; g_return_if_fail (EPHY_IS_NODE_VIEW (view)); window = gtk_widget_get_toplevel (GTK_WIDGET (view)); if (window != NULL && window->window != NULL) { /* nice busy cursor */ GdkCursor *cursor; cursor = gdk_cursor_new (GDK_WATCH); gdk_window_set_cursor (window->window, cursor); gdk_cursor_unref (cursor); gdk_flush (); gdk_window_set_cursor (window->window, NULL); /* no flush: this will cause the cursor to be reset * only when the UI is free again */ } } static void ephy_node_view_selection_changed_cb (GtkTreeSelection *selection, EphyNodeView *view) { GList *list; EphyNode *node = NULL; list = ephy_node_view_get_selection (view); if (list) { node = list->data; } g_list_free (list); g_signal_emit (G_OBJECT (view), ephy_node_view_signals[NODE_SELECTED], 0, node); } static void ephy_node_view_row_activated_cb (GtkTreeView *treeview, GtkTreePath *path, GtkTreeViewColumn *column, EphyNodeView *view) { GtkTreeIter iter, iter2; EphyNode *node; gtk_tree_model_get_iter (view->priv->sortmodel, &iter, path); gtk_tree_model_sort_convert_iter_to_child_iter (GTK_TREE_MODEL_SORT (view->priv->sortmodel), &iter2, &iter); egg_tree_model_filter_convert_iter_to_child_iter (EGG_TREE_MODEL_FILTER (view->priv->filtermodel), &iter, &iter2); node = ephy_tree_model_node_node_from_iter (view->priv->nodemodel, &iter); g_signal_emit (G_OBJECT (view), ephy_node_view_signals[NODE_ACTIVATED], 0, node); } static gboolean ephy_node_view_select_node_by_key (EphyNodeView *view, GdkEventKey *event) { GtkTreeIter iter, last_iter; GtkTreePath *path; GValue value = {0, }; gchar *string; gchar *event_string; gboolean found = FALSE; gchar outbuf[6]; gint length; length = g_unichar_to_utf8 (gdk_keyval_to_unicode (event->keyval), outbuf); event_string = g_utf8_casefold (outbuf, length); if (!gtk_tree_model_get_iter_first (view->priv->sortmodel, &iter)) { g_free (event_string); return FALSE; } do { last_iter = iter; gtk_tree_model_get_value (view->priv->sortmodel, &iter, view->priv->searchable_data_column, &value); string = g_utf8_casefold (g_value_get_string (&value), -1); g_utf8_strncpy (string, string, 1); found = (g_utf8_collate (string, event_string) >= 0); g_free (string); g_value_unset (&value); } while (!found && gtk_tree_model_iter_next (view->priv->sortmodel, &iter)); if (!found) { iter = last_iter; } path = gtk_tree_model_get_path (view->priv->sortmodel, &iter); gtk_tree_view_set_cursor (GTK_TREE_VIEW (view), path, NULL, FALSE); gtk_tree_path_free (path); g_free (event_string); return TRUE; } static gboolean ephy_node_view_key_press_cb (GtkTreeView *treeview, GdkEventKey *event, EphyNodeView *view) { if ((event->state & GDK_SHIFT_MASK) && (event->keyval == GDK_F10)) { g_signal_emit (G_OBJECT (view), ephy_node_view_signals[SHOW_POPUP], 0); return TRUE; } else if (view->priv->searchable_data_column != -1 && gdk_keyval_to_unicode (event->keyval)) { return ephy_node_view_select_node_by_key (view, event); } return FALSE; } static gboolean ephy_node_view_button_press_cb (GtkWidget *treeview, GdkEventButton *event, EphyNodeView *view) { GtkTreePath *path = NULL; GtkTreeSelection *selection; if (event->button != 3) return FALSE; if (gtk_tree_view_get_path_at_pos (GTK_TREE_VIEW (treeview), event->x, event->y, &path, NULL, NULL, NULL)) { selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (treeview)); if (!gtk_tree_selection_path_is_selected (selection, path)) { GTK_WIDGET_CLASS (parent_class)->button_press_event (treeview, event); } else if (!gtk_widget_is_focus (GTK_WIDGET (treeview))) { gtk_widget_grab_focus (GTK_WIDGET (treeview)); } g_signal_emit (G_OBJECT (view), ephy_node_view_signals[SHOW_POPUP], 0); gtk_tree_path_free (path); } return TRUE; } static void ephy_node_view_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { EphyNodeView *view = EPHY_NODE_VIEW (object); switch (prop_id) { case PROP_ROOT: view->priv->root = g_value_get_pointer (value); break; case PROP_FILTER: view->priv->filter = g_value_get_object (value); if (view->priv->filter != NULL) { g_signal_connect_object (G_OBJECT (view->priv->filter), "changed", G_CALLBACK (filter_changed_cb), G_OBJECT (view), 0); } break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void ephy_node_view_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { EphyNodeView *view = EPHY_NODE_VIEW (object); switch (prop_id) { case PROP_ROOT: g_value_set_pointer (value, view->priv->root); break; case PROP_FILTER: g_value_set_object (value, view->priv->filter); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void node_from_sort_iter_cb (EphyTreeModelSort *model, GtkTreeIter *iter, void **node, EphyNodeView *view) { GtkTreeIter filter_iter, node_iter; gtk_tree_model_sort_convert_iter_to_child_iter (GTK_TREE_MODEL_SORT (model), &filter_iter, iter); egg_tree_model_filter_convert_iter_to_child_iter (EGG_TREE_MODEL_FILTER (view->priv->filtermodel), &node_iter, &filter_iter); *node = ephy_tree_model_node_node_from_iter (EPHY_TREE_MODEL_NODE (view->priv->nodemodel), &node_iter); } static void ephy_node_view_construct (EphyNodeView *view) { GtkTreeSelection *selection; view->priv->nodemodel = ephy_tree_model_node_new (view->priv->root, view->priv->filter); view->priv->filtermodel = egg_tree_model_filter_new (GTK_TREE_MODEL (view->priv->nodemodel), NULL); egg_tree_model_filter_set_visible_column (EGG_TREE_MODEL_FILTER (view->priv->filtermodel), EPHY_TREE_MODEL_NODE_COL_VISIBLE); view->priv->sortmodel = ephy_tree_model_sort_new (view->priv->filtermodel); g_signal_connect_object (G_OBJECT (view->priv->sortmodel), "node_from_iter", G_CALLBACK (node_from_sort_iter_cb), view, 0); gtk_tree_view_set_model (GTK_TREE_VIEW (view), GTK_TREE_MODEL (view->priv->sortmodel)); g_signal_connect_object (G_OBJECT (view), "button_press_event", G_CALLBACK (ephy_node_view_button_press_cb), view, 0); g_signal_connect_after (G_OBJECT (view), "key_press_event", G_CALLBACK (ephy_node_view_key_press_cb), view); g_signal_connect_object (G_OBJECT (view), "row_activated", G_CALLBACK (ephy_node_view_row_activated_cb), view, 0); selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view)); gtk_tree_selection_set_mode (selection, GTK_SELECTION_MULTIPLE); g_signal_connect_object (G_OBJECT (selection), "changed", G_CALLBACK (ephy_node_view_selection_changed_cb), view, 0); } GtkWidget * ephy_node_view_new (EphyNode *root, EphyNodeFilter *filter) { EphyNodeView *view; view = EPHY_NODE_VIEW (g_object_new (EPHY_TYPE_NODE_VIEW, "filter", filter, "root", root, NULL)); ephy_node_view_construct (view); g_return_val_if_fail (view->priv != NULL, NULL); return GTK_WIDGET (view); } static void cell_renderer_edited (GtkCellRendererText *cell, const char *path_str, const char *new_text, EphyNodeView *view) { GValue value = { 0, }; GtkTreePath *path; GtkTreeIter iter, iter2; EphyNode *node; view->priv->editing = FALSE; g_object_set (G_OBJECT (view->priv->editable_renderer), "editable", FALSE, NULL); path = gtk_tree_path_new_from_string (path_str); gtk_tree_model_get_iter (view->priv->sortmodel, &iter, path); gtk_tree_model_sort_convert_iter_to_child_iter (GTK_TREE_MODEL_SORT (view->priv->sortmodel), &iter2, &iter); egg_tree_model_filter_convert_iter_to_child_iter (EGG_TREE_MODEL_FILTER (view->priv->filtermodel), &iter, &iter2); node = ephy_tree_model_node_node_from_iter (view->priv->nodemodel, &iter); g_value_init (&value, G_TYPE_STRING); g_value_set_string (&value, new_text); ephy_node_set_property (node, view->priv->editable_property, &value); g_value_unset (&value); } static int ephy_node_view_sort_func (GtkTreeModel *model, GtkTreeIter *a, GtkTreeIter *b, gpointer user_data) { GList *order; GList *l; int retval = 0; g_return_val_if_fail (model != NULL, 0); g_return_val_if_fail (user_data != NULL, 0); order = (GList *) user_data; for (l = order; l != NULL && retval == 0; l = g_list_next (l)) { EphyTreeModelNodeColumn column = GPOINTER_TO_INT (l->data); GType type = gtk_tree_model_get_column_type (model, column); GValue a_value = {0, }; GValue b_value = {0, }; gchar *stra, *strb; gtk_tree_model_get_value (model, a, column, &a_value); gtk_tree_model_get_value (model, b, column, &b_value); switch (G_TYPE_FUNDAMENTAL (type)) { case G_TYPE_STRING: /* FIXME: this is horribly inefficient */ stra = g_utf8_casefold (g_value_get_string (&a_value), -1); strb = g_utf8_casefold (g_value_get_string (&b_value), -1); g_return_val_if_fail (stra != NULL && strb != NULL, 0); retval = g_utf8_collate (stra, strb); g_free (stra); g_free (strb); break; case G_TYPE_INT: if (g_value_get_int (&a_value) < g_value_get_int (&b_value)) retval = -1; else if (g_value_get_int (&a_value) == g_value_get_int (&b_value)) retval = 0; else retval = 1; break; case G_TYPE_BOOLEAN: if (g_value_get_boolean (&a_value) < g_value_get_boolean (&b_value)) retval = -1; else if (g_value_get_boolean (&a_value) == g_value_get_boolean (&b_value)) retval = 0; else retval = 1; break; default: g_warning ("Attempting to sort on invalid type %s\n", g_type_name (type)); break; } g_value_unset (&a_value); g_value_unset (&b_value); } return retval; } static void provide_priority (EphyNode *node, GValue *value, EphyNodeView *view) { int priority; g_value_init (value, G_TYPE_INT); priority = ephy_node_get_property_int (node, view->priv->priority_prop_id); if (priority == EPHY_NODE_VIEW_ALL_PRIORITY || priority == EPHY_NODE_VIEW_SPECIAL_PRIORITY) g_value_set_int (value, priority); else g_value_set_int (value, EPHY_NODE_VIEW_NORMAL_PRIORITY); } static void provide_text_weight (EphyNode *node, GValue *value, EphyNodeView *view) { int priority; g_value_init (value, G_TYPE_INT); priority = ephy_node_get_property_int (node, view->priv->priority_prop_id); if (priority == EPHY_NODE_VIEW_ALL_PRIORITY || priority == EPHY_NODE_VIEW_SPECIAL_PRIORITY) { g_value_set_int (value, PANGO_WEIGHT_BOLD); } else { g_value_set_int (value, PANGO_WEIGHT_NORMAL); } } GtkTreeViewColumn * ephy_node_view_add_column (EphyNodeView *view, const char *title, GType value_type, int prop_id, int priority_prop_id, EphyNodeViewFlags flags, EphyTreeModelNodeValueFunc icon_func) { GtkTreeViewColumn *gcolumn; GtkCellRenderer *renderer; int column; int icon_column; g_return_val_if_fail (!(flags & EPHY_NODE_VIEW_EDITABLE) || view->priv->editable_renderer == NULL, NULL); column = ephy_tree_model_node_add_prop_column (view->priv->nodemodel, value_type, prop_id); gcolumn = (GtkTreeViewColumn *) gtk_tree_view_column_new (); if (icon_func) { icon_column = ephy_tree_model_node_add_func_column (view->priv->nodemodel, GDK_TYPE_PIXBUF, icon_func, NULL); renderer = gtk_cell_renderer_pixbuf_new (); gtk_tree_view_column_pack_start (gcolumn, renderer, FALSE); gtk_tree_view_column_set_attributes (gcolumn, renderer, "pixbuf", icon_column, NULL); } renderer = gtk_cell_renderer_text_new (); if (flags & EPHY_NODE_VIEW_EDITABLE) { view->priv->editable_renderer = renderer; view->priv->editable_column = gcolumn; view->priv->editable_node_column = column; view->priv->editable_property = prop_id; g_signal_connect (renderer, "edited", G_CALLBACK (cell_renderer_edited), view); } if ((flags & EPHY_NODE_VIEW_SEARCHABLE) && (view->priv->searchable_data_column == -1)) { view->priv->searchable_data_column = column; } gtk_tree_view_column_pack_start (gcolumn, renderer, TRUE); gtk_tree_view_column_set_attributes (gcolumn, renderer, "text", column, NULL); if (priority_prop_id >= 0) { int wcol; view->priv->priority_prop_id = priority_prop_id; wcol = ephy_tree_model_node_add_func_column (view->priv->nodemodel, G_TYPE_INT, (EphyTreeModelNodeValueFunc) provide_text_weight, view); gtk_tree_view_column_add_attribute (gcolumn, renderer, "weight", wcol); } gtk_tree_view_column_set_title (gcolumn, title); gtk_tree_view_append_column (GTK_TREE_VIEW (view), gcolumn); if (flags & EPHY_NODE_VIEW_USER_SORT) { GList *order = NULL; order = g_list_append (order, GINT_TO_POINTER (column)); gtk_tree_sortable_set_sort_func (GTK_TREE_SORTABLE (view->priv->sortmodel), column, ephy_node_view_sort_func, order, (GtkDestroyNotify)g_list_free); gtk_tree_view_column_set_sort_column_id (gcolumn, column); } else if (flags & EPHY_NODE_VIEW_AUTO_SORT) { int scol; GList *order = NULL; if (priority_prop_id >= 0) { scol = ephy_tree_model_node_add_func_column (view->priv->nodemodel, G_TYPE_INT, (EphyTreeModelNodeValueFunc) provide_priority, view); order = g_list_append (order, GINT_TO_POINTER (scol)); } order = g_list_append (order, GINT_TO_POINTER (column)); gtk_tree_sortable_set_default_sort_func (GTK_TREE_SORTABLE (view->priv->sortmodel), ephy_node_view_sort_func, order, (GtkDestroyNotify)g_list_free); gtk_tree_sortable_set_sort_column_id (GTK_TREE_SORTABLE (view->priv->sortmodel), GTK_TREE_SORTABLE_DEFAULT_SORT_COLUMN_ID, GTK_SORT_ASCENDING); } return gcolumn; } static void ephy_node_view_init (EphyNodeView *view) { view->priv = g_new0 (EphyNodeViewPrivate, 1); view->priv->editable_renderer = NULL; view->priv->editing = TRUE; view->priv->searchable_data_column = -1; gtk_tree_view_set_enable_search (GTK_TREE_VIEW (view), FALSE); } static void get_selection (GtkTreeModel *model, GtkTreePath *path, GtkTreeIter *iter, void **data) { GtkTreeModelSort *sortmodel = GTK_TREE_MODEL_SORT (model); EggTreeModelFilter *filtermodel = EGG_TREE_MODEL_FILTER (sortmodel->child_model); EphyTreeModelNode *nodemodel = EPHY_TREE_MODEL_NODE (filtermodel->child_model); GList **list = (GList **) data; GtkTreeIter *iter2 = gtk_tree_iter_copy (iter); GtkTreeIter iter3; GtkTreeIter iter4; EphyNode *node; gtk_tree_model_sort_convert_iter_to_child_iter (GTK_TREE_MODEL_SORT (model), &iter3, iter2); egg_tree_model_filter_convert_iter_to_child_iter (filtermodel, &iter4, &iter3); node = ephy_tree_model_node_node_from_iter (nodemodel, &iter4); gtk_tree_iter_free (iter2); *list = g_list_prepend (*list, node); } GList * ephy_node_view_get_selection (EphyNodeView *view) { GList *list = NULL; GtkTreeSelection *selection; selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view)); gtk_tree_selection_selected_foreach (selection, (GtkTreeSelectionForeachFunc) get_selection, (gpointer) &list); return list; } void ephy_node_view_remove (EphyNodeView *view) { GList *list; EphyNode *node; GtkTreeIter iter, iter2; GtkTreePath *path; GtkTreeRowReference *row_ref = NULL; /* Before removing we try to get a reference to the next node in the view. If that is * not available we try with the previous one, and if that is absent too, * we will not select anything (which equals to select the topic "All") */ list = ephy_node_view_get_selection (view); g_return_if_fail (list != NULL); node = g_list_first (list)->data; ephy_tree_model_node_iter_from_node (EPHY_TREE_MODEL_NODE (view->priv->nodemodel), node, &iter); egg_tree_model_filter_convert_child_iter_to_iter (EGG_TREE_MODEL_FILTER (view->priv->filtermodel), &iter2, &iter); gtk_tree_model_sort_convert_child_iter_to_iter (GTK_TREE_MODEL_SORT (view->priv->sortmodel), &iter, &iter2); iter2 = iter; if (gtk_tree_model_iter_next (GTK_TREE_MODEL (view->priv->sortmodel), &iter)) { path = gtk_tree_model_get_path (GTK_TREE_MODEL (view->priv->sortmodel), &iter); row_ref = gtk_tree_row_reference_new (GTK_TREE_MODEL (view->priv->sortmodel), path); } else { path = gtk_tree_model_get_path (GTK_TREE_MODEL (view->priv->sortmodel), &iter2); if (gtk_tree_path_prev (path)) { row_ref = gtk_tree_row_reference_new (GTK_TREE_MODEL (view->priv->sortmodel), path); } } gtk_tree_path_free (path); for (; list != NULL; list = list->next) { ephy_node_unref (list->data); } g_list_free (list); /* Select the "next" node */ if (row_ref != NULL) { path = gtk_tree_row_reference_get_path (row_ref); gtk_tree_view_set_cursor (GTK_TREE_VIEW (view), path, NULL, FALSE); gtk_tree_row_reference_free (row_ref); gtk_tree_path_free (path); } } void ephy_node_view_select_node (EphyNodeView *view, EphyNode *node) { GtkTreeIter iter, iter2; GValue val = { 0, }; gboolean visible; g_return_if_fail (node != NULL); ephy_tree_model_node_iter_from_node (EPHY_TREE_MODEL_NODE (view->priv->nodemodel), node, &iter); gtk_tree_model_get_value (GTK_TREE_MODEL (view->priv->nodemodel), &iter, EPHY_TREE_MODEL_NODE_COL_VISIBLE, &val); visible = g_value_get_boolean (&val); g_value_unset (&val); if (visible == FALSE) return; egg_tree_model_filter_convert_child_iter_to_iter (EGG_TREE_MODEL_FILTER (view->priv->filtermodel), &iter2, &iter); gtk_tree_model_sort_convert_child_iter_to_iter (GTK_TREE_MODEL_SORT (view->priv->sortmodel), &iter, &iter2); gtk_tree_selection_select_iter (gtk_tree_view_get_selection (GTK_TREE_VIEW (view)), &iter); } void ephy_node_view_enable_drag_source (EphyNodeView *view, GtkTargetEntry *types, int n_types, guint prop_id) { GtkWidget *treeview; g_return_if_fail (view != NULL); treeview = GTK_WIDGET (view); egg_tree_multi_drag_add_drag_support (GTK_TREE_VIEW (treeview)); gtk_tree_view_enable_model_drag_source (GTK_TREE_VIEW (treeview), GDK_BUTTON1_MASK, types, n_types, GDK_ACTION_COPY); ephy_tree_model_sort_set_drag_property (EPHY_TREE_MODEL_SORT (view->priv->sortmodel), prop_id); } void ephy_node_view_edit (EphyNodeView *view) { GtkTreeSelection *selection; GList *rows; GtkTreeModel *model; g_return_if_fail (view->priv->editable_renderer != NULL); selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view)); rows = gtk_tree_selection_get_selected_rows (selection, &model); if (rows == NULL) return; g_object_set (G_OBJECT (view->priv->editable_renderer), "editable", TRUE, NULL); gtk_tree_view_set_cursor (GTK_TREE_VIEW (view), (GtkTreePath *)rows->data, view->priv->editable_column, TRUE); view->priv->editing = TRUE; g_list_foreach (rows, (GFunc)gtk_tree_path_free, NULL); g_list_free (rows); } gboolean ephy_node_view_is_target (EphyNodeView *view) { return gtk_widget_is_focus (GTK_WIDGET (view)); } gboolean ephy_node_view_has_selection (EphyNodeView *view, gboolean *multiple) { GtkTreeSelection *selection; int rows; selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view)); rows = gtk_tree_selection_count_selected_rows (selection); if (multiple) { *multiple = rows > 1; } return rows > 0; }