/* -*- Mode: C; tab-width: 8; indent-tabs-mode: t; c-basic-offset: 8 -*- */
/*
* Copyright © 2002 Jorn Baayen <jorn@nl.linux.org>
*
* 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$
*/
#include "config.h"
#include <gtk/gtktreeselection.h>
#include <gtk/gtktreeviewcolumn.h>
#include <gtk/gtkcellrenderertext.h>
#include <gtk/gtkcellrendererpixbuf.h>
#include <gtk/gtkcellrenderertoggle.h>
#include <gtk/gtktreemodelfilter.h>
#include <gtk/gtkwindow.h>
#include <gtk/gtkmain.h>
#include <gdk/gdkkeysyms.h>
#include "ephy-node-view.h"
#include "ephy-tree-model-sort.h"
#include "eggtreemultidnd.h"
#include "ephy-dnd.h"
#include "ephy-gui.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);
#define EPHY_NODE_VIEW_GET_PRIVATE(object)(G_TYPE_INSTANCE_GET_PRIVATE ((object), EPHY_TYPE_NODE_VIEW, EphyNodeViewPrivate))
struct _EphyNodeViewPrivate
{
EphyNode *root;
EphyTreeModelNode *nodemodel;
GtkTreeModel *filtermodel;
GtkTreeModel *sortmodel;
GtkCellRenderer *editable_renderer;
GtkTreeViewColumn *editable_column;
int editable_node_column;
int toggle_column;
EphyNodeFilter *filter;
GtkTargetList *drag_targets;
int sort_column;
GtkSortType sort_type;
guint priority_prop_id;
int priority_column;
EphyNode *edited_node;
gboolean remove_if_cancelled;
int editable_property;
gboolean drag_started;
int drag_button;
int drag_x;
int drag_y;
GtkTargetList *source_target_list;
gboolean drop_occurred;
gboolean have_drag_data;
GtkSelectionData *drag_data;
guint scroll_id;
guint changing_selection : 1;
};
enum
{
NODE_TOGGLED,
NODE_ACTIVATED,
NODE_SELECTED,
NODE_DROPPED,
NODE_MIDDLE_CLICKED,
LAST_SIGNAL
};
enum
{
PROP_0,
PROP_ROOT,
PROP_FILTER
};
#define AUTO_SCROLL_MARGIN 20
static GObjectClass *parent_class = NULL;
static guint ephy_node_view_signals[LAST_SIGNAL] = { 0 };
GType
ephy_node_view_get_type (void)
{
static GType type = 0;
if (G_UNLIKELY (type == 0))
{
const GTypeInfo our_info =
{
sizeof (EphyNodeViewClass),
NULL,
NULL,
(GClassInitFunc) ephy_node_view_class_init,
NULL,
NULL,
sizeof (EphyNodeView),
0,
(GInstanceInitFunc) ephy_node_view_init
};
type = g_type_register_static (GTK_TYPE_TREE_VIEW,
"EphyNodeView",
&our_info, 0);
}
return type;
}
static void
ephy_node_view_finalize (GObject *object)
{
EphyNodeView *view = EPHY_NODE_VIEW (object);
g_object_unref (G_OBJECT (view->priv->sortmodel));
g_object_unref (G_OBJECT (view->priv->filtermodel));
g_object_unref (G_OBJECT (view->priv->nodemodel));
if (view->priv->source_target_list)
{
gtk_target_list_unref (view->priv->source_target_list);
}
if (view->priv->drag_targets)
{
gtk_target_list_unref (view->priv->drag_targets);
}
G_OBJECT_CLASS (parent_class)->finalize (object);
}
static EphyNode *
get_node_from_path (EphyNodeView *view, GtkTreePath *path)
{
EphyNode *node;
GtkTreeIter iter, iter2, iter3;
if (path == NULL) return NULL;
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);
if (iter2.stamp == 0) {
return NULL;
}
gtk_tree_model_filter_convert_iter_to_child_iter
(GTK_TREE_MODEL_FILTER (view->priv->filtermodel), &iter3, &iter2);
node = ephy_tree_model_node_node_from_iter (view->priv->nodemodel, &iter3);
return node;
}
static void
gtk_tree_view_vertical_autoscroll (GtkTreeView *tree_view)
{
GdkRectangle visible_rect;
GtkAdjustment *vadjustment;
GdkWindow *window;
int y;
int offset;
float value;
window = gtk_tree_view_get_bin_window (tree_view);
vadjustment = gtk_tree_view_get_vadjustment (tree_view);
gdk_window_get_pointer (window, NULL, &y, NULL);
y += vadjustment->value;
gtk_tree_view_get_visible_rect (tree_view, &visible_rect);
offset = y - (visible_rect.y + 2 * AUTO_SCROLL_MARGIN);
if (offset > 0)
{
offset = y - (visible_rect.y + visible_rect.height - 2 * AUTO_SCROLL_MARGIN);
if (offset < 0)
{
return;
}
}
value = CLAMP (vadjustment->value + offset, 0.0,
vadjustment->upper - vadjustment->page_size);
gtk_adjustment_set_value (vadjustment, value);
}
static int
scroll_timeout (gpointer data)
{
GtkTreeView *tree_view = GTK_TREE_VIEW (data);
gtk_tree_view_vertical_autoscroll (tree_view);
return TRUE;
}
static void
remove_scroll_timeout (EphyNodeView *view)
{
if (view->priv->scroll_id)
{
g_source_remove (view->priv->scroll_id);
view->priv->scroll_id = 0;
}
}
static void
set_drag_dest_row (EphyNodeView *view,
GtkTreePath *path)
{
if (path)
{
gtk_tree_view_set_drag_dest_row
(GTK_TREE_VIEW (view),
path,
GTK_TREE_VIEW_DROP_INTO_OR_BEFORE);
}
else
{
gtk_tree_view_set_drag_dest_row (GTK_TREE_VIEW (view),
NULL,
0);
}
}
static void
clear_drag_dest_row (EphyNodeView *view)
{
gtk_tree_view_set_drag_dest_row (GTK_TREE_VIEW (view), NULL, 0);
}
static void
get_drag_data (EphyNodeView *view,
GdkDragContext *context,
guint32 time)
{
GdkAtom target;
target = gtk_drag_dest_find_target (GTK_WIDGET (view),
context,
NULL);
gtk_drag_get_data (GTK_WIDGET (view),
context, target, time);
}
static void
free_drag_data (EphyNodeView *view)
{
view->priv->have_drag_data = FALSE;
if (view->priv->drag_data)
{
gtk_selection_data_free (view->priv->drag_data);
view->priv->drag_data = NULL;
}
}
static gboolean
drag_motion_cb (GtkWidget *widget,
GdkDragContext *context,
int x,
int y,
guint32 time,
EphyNodeView *view)
{
EphyNode *node;
GdkAtom target;
GtkTreePath *path;
GtkTreeViewDropPosition pos;
guint action = 0;
int priority;
gtk_tree_view_get_dest_row_at_pos (GTK_TREE_VIEW (widget),
x, y, &path, &pos);
if (!view->priv->have_drag_data)
{
get_drag_data (view, context, time);
}
target = gtk_drag_dest_find_target (widget, context, NULL);
node = get_node_from_path (view, path);
if (target != GDK_NONE && node != NULL)
{
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 &&
ephy_node_get_is_drag_source (node))
{
action = context->suggested_action;
}
}
if (action)
{
set_drag_dest_row (view, path);
}
else
{
clear_drag_dest_row (view);
}
if (path)
{
gtk_tree_path_free (path);
}
if (view->priv->scroll_id == 0)
{
view->priv->scroll_id =
g_timeout_add (150,
scroll_timeout,
GTK_TREE_VIEW (view));
}
gdk_drag_status (context, action, time);
return TRUE;
}
static void
drag_leave_cb (GtkWidget *widget,
GdkDragContext *context,
guint32 time,
EphyNodeView *view)
{
clear_drag_dest_row (view);
free_drag_data (view);
remove_scroll_timeout (view);
}
static void
drag_data_received_cb (GtkWidget *widget,
GdkDragContext *context,
int x,
int y,
GtkSelectionData *selection_data,
guint info,
guint32 time,
EphyNodeView *view)
{
GtkTreeViewDropPosition pos;
/* x and y here are valid only on drop ! */
if (selection_data->length <= 0 || selection_data->data == NULL)
{
return;
}
/* appease GtkTreeView by preventing its drag_data_receive
* from being called */
g_signal_stop_emission_by_name (view, "drag_data_received");
if (!view->priv->have_drag_data)
{
view->priv->have_drag_data = TRUE;
view->priv->drag_data =
gtk_selection_data_copy (selection_data);
}
if (view->priv->drop_occurred)
{
EphyNode *node;
char **uris;
gboolean success = FALSE;
GtkTreePath *path;
if (gtk_tree_view_get_dest_row_at_pos
(GTK_TREE_VIEW (widget), x, y, &path, &pos) == FALSE)
{
return;
}
node = get_node_from_path (view, path);
if (node == NULL) return;
uris = gtk_selection_data_get_uris (selection_data);
if (uris != NULL && ephy_node_get_is_drag_dest (node))
{
/* FIXME fill success */
g_signal_emit (G_OBJECT (view),
ephy_node_view_signals[NODE_DROPPED], 0,
node, uris);
g_strfreev (uris);
}
view->priv->drop_occurred = FALSE;
free_drag_data (view);
gtk_drag_finish (context, success, FALSE, time);
if (path)
{
gtk_tree_path_free (path);
}
}
}
static gboolean
drag_drop_cb (GtkWidget *widget,
GdkDragContext *context,
int x,
int y,
guint32 time,
EphyNodeView *view)
{
view->priv->drop_occurred = TRUE;
get_drag_data (view, context, time);
remove_scroll_timeout (view);
clear_drag_dest_row (view);
return TRUE;
}
void
ephy_node_view_enable_drag_dest (EphyNodeView *view,
const GtkTargetEntry *types,
int n_types)
{
GtkWidget *treeview;
g_return_if_fail (view != NULL);
treeview = GTK_WIDGET (view);
gtk_drag_dest_set (GTK_WIDGET (treeview),
0, 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);
g_signal_connect (treeview, "drag_leave",
G_CALLBACK (drag_leave_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 */
}
gtk_tree_model_filter_refilter
(GTK_TREE_MODEL_FILTER (view->priv->filtermodel));
}
static void
ephy_node_view_selection_changed_cb (GtkTreeSelection *selection,
EphyNodeView *view)
{
EphyNodeViewPrivate *priv = view->priv;
GList *list;
EphyNode *node = NULL;
/* Work around bug #346662 */
if (priv->changing_selection) return;
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);
gtk_tree_model_filter_convert_iter_to_child_iter
(GTK_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 void
path_toggled (GtkTreeModel *dummy_model, GtkTreePath *path,
GtkTreeIter *dummy, gpointer data)
{
EphyNodeView *view = EPHY_NODE_VIEW (data);
gboolean checked;
EphyNode *node;
GtkTreeIter iter, iter2;
GValue value = {0, };
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);
gtk_tree_model_filter_convert_iter_to_child_iter
(GTK_TREE_MODEL_FILTER (view->priv->filtermodel), &iter, &iter2);
node = ephy_tree_model_node_node_from_iter (view->priv->nodemodel, &iter);
gtk_tree_model_get_value (GTK_TREE_MODEL (view->priv->nodemodel), &iter,
view->priv->toggle_column, &value);
checked = !g_value_get_boolean (&value);
g_signal_emit (G_OBJECT (view), ephy_node_view_signals[NODE_TOGGLED], 0,
node, checked);
}
static EphyNode *
process_middle_click (GtkTreePath *path,
EphyNodeView *view)
{
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);
gtk_tree_model_filter_convert_iter_to_child_iter
(GTK_TREE_MODEL_FILTER (view->priv->filtermodel), &iter, &iter2);
node = ephy_tree_model_node_node_from_iter (view->priv->nodemodel, &iter);
return node;
}
static gboolean
ephy_node_view_key_press_cb (GtkTreeView *treeview,
GdkEventKey *event,
EphyNodeView *view)
{
gboolean handled = FALSE;
if (event->keyval == GDK_space ||
event->keyval == GDK_Return ||
event->keyval == GDK_KP_Enter ||
event->keyval == GDK_ISO_Enter)
{
if (view->priv->toggle_column >= 0)
{
GtkTreeSelection *selection;
selection = gtk_tree_view_get_selection (treeview);
gtk_tree_selection_selected_foreach
(selection, path_toggled, view);
handled = TRUE;
}
}
return handled;
}
static void
selection_foreach (GtkTreeModel *model,
GtkTreePath *path,
GtkTreeIter *iter,
gpointer data)
{
GList **list;
list = (GList**)data;
*list = g_list_prepend (*list,
gtk_tree_row_reference_new (model, path));
}
static GList *
get_selection_refs (GtkTreeView *tree_view)
{
GtkTreeSelection *selection;
GList *ref_list = NULL;
selection = gtk_tree_view_get_selection (tree_view);
gtk_tree_selection_selected_foreach (selection,
selection_foreach,
&ref_list);
ref_list = g_list_reverse (ref_list);
return ref_list;
}
static void
ref_list_free (GList *ref_list)
{
g_list_foreach (ref_list, (GFunc) gtk_tree_row_reference_free, NULL);
g_list_free (ref_list);
}
static void
stop_drag_check (EphyNodeView *view)
{
view->priv->drag_button = 0;
}
static gboolean
button_event_modifies_selection (GdkEventButton *event)
{
return (event->state & (GDK_CONTROL_MASK | GDK_SHIFT_MASK)) != 0;
}
static void
did_not_drag (EphyNodeView *view,
GdkEventButton *event)
{
GtkTreeView *tree_view;
GtkTreeSelection *selection;
GtkTreePath *path;
tree_view = GTK_TREE_VIEW (view);
selection = gtk_tree_view_get_selection (tree_view);
if (gtk_tree_view_get_path_at_pos (tree_view, event->x, event->y,
&path, NULL, NULL, NULL))
{
if((event->button == 1 || event->button == 2) &&
gtk_tree_selection_path_is_selected (selection, path) &&
!button_event_modifies_selection (event))
{
if (gtk_tree_selection_get_mode (selection) == GTK_SELECTION_MULTIPLE)
{
gtk_tree_selection_unselect_all (selection);
}
gtk_tree_selection_select_path (selection, path);
}
gtk_tree_path_free (path);
}
}
typedef struct
{
EphyNodeView *view;
gboolean result;
}
ForeachData;
static void
check_node_is_drag_source (GtkTreeModel *model,
GtkTreePath *path,
GtkTreeIter *iter,
ForeachData *data)
{
EphyNode *node;
node = get_node_from_path (data->view, path);
data->result = data->result &&
node != NULL &&
ephy_node_get_is_drag_source (node);
}
static gboolean
can_drag_selection (EphyNodeView *view)
{
GtkTreeView *tree_view = GTK_TREE_VIEW (view);
GtkTreeSelection *selection;
ForeachData data = { view, TRUE };
selection = gtk_tree_view_get_selection (tree_view);
gtk_tree_selection_selected_foreach (selection,
(GtkTreeSelectionForeachFunc) check_node_is_drag_source,
&data);
return data.result;
}
static void
drag_data_get_cb (GtkWidget *widget,
GdkDragContext *context,
GtkSelectionData *selection_data,
guint info,
guint time)
{
GtkTreeView *tree_view;
GtkTreeModel *model;
GList *ref_list;
tree_view = GTK_TREE_VIEW (widget);
model = gtk_tree_view_get_model (tree_view);
g_return_if_fail (model != NULL);
ref_list = g_object_get_data (G_OBJECT (context), "drag-info");
if (ref_list == NULL)
{
return;
}
if (EGG_IS_TREE_MULTI_DRAG_SOURCE (model))
{
egg_tree_multi_drag_source_drag_data_get (EGG_TREE_MULTI_DRAG_SOURCE (model),
ref_list,
selection_data);
}
}
static gboolean
button_release_cb (GtkWidget *widget,
GdkEventButton *event,
EphyNodeView *view)
{
if (event->button == view->priv->drag_button)
{
stop_drag_check (view);
if (!view->priv->drag_started)
{
did_not_drag (view, event);
return TRUE;
}
view->priv->drag_started = FALSE;
}
return FALSE;
}
static gboolean
motion_notify_cb (GtkWidget *widget,
GdkEventMotion *event,
EphyNodeView *view)
{
GdkDragContext *context;
GList *ref_list;
if (event->window != gtk_tree_view_get_bin_window (GTK_TREE_VIEW (widget)))
{
return FALSE;
}
if (view->priv->drag_button != 0)
{
if (gtk_drag_check_threshold (widget, view->priv->drag_x,
view->priv->drag_y, event->x,
event->y)
&& can_drag_selection (view))
{
context = gtk_drag_begin
(widget, view->priv->source_target_list,
GDK_ACTION_ASK | GDK_ACTION_COPY | GDK_ACTION_LINK,
view->priv->drag_button,
(GdkEvent*)event);
stop_drag_check (view);
view->priv->drag_started = TRUE;
ref_list = get_selection_refs (GTK_TREE_VIEW (widget));
g_object_set_data_full (G_OBJECT (context),
"drag-info",
ref_list,
(GDestroyNotify)ref_list_free);
gtk_drag_set_icon_default (context);
}
}
return TRUE;
}
static gboolean
ephy_node_view_button_press_cb (GtkWidget *treeview,
GdkEventButton *event,
EphyNodeView *view)
{
GtkTreePath *path = NULL;
GtkTreeSelection *selection;
gboolean call_parent = TRUE, path_is_selected;
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (treeview));
if (event->window != gtk_tree_view_get_bin_window (GTK_TREE_VIEW (treeview)))
{
return GTK_WIDGET_CLASS (parent_class)->button_press_event (treeview, event);
}
if (gtk_tree_view_get_path_at_pos (GTK_TREE_VIEW (treeview),
event->x,
event->y,
&path,
NULL, NULL, NULL))
{
path_is_selected = gtk_tree_selection_path_is_selected (selection, path);
if (!gtk_widget_is_focus (GTK_WIDGET (treeview)))
{
gtk_widget_grab_focus (GTK_WIDGET (treeview));
}
if (event->button == 3 && path_is_selected)
{
call_parent = FALSE;
}
if(!button_event_modifies_selection (event) &&
event->button == 1 && path_is_selected &&
gtk_tree_selection_count_selected_rows (selection) > 1)
{
call_parent = FALSE;
}
if (call_parent)
{
GTK_WIDGET_CLASS (parent_class)->button_press_event (treeview, event);
}
if (event->button == 3)
{
gboolean retval;
g_signal_emit_by_name (view, "popup_menu", &retval);
}
else if (event->button == 2)
{
EphyNode *clicked_node;
clicked_node = process_middle_click (path, view);
g_signal_emit (G_OBJECT (view),
ephy_node_view_signals[NODE_MIDDLE_CLICKED], 0, clicked_node);
}
else if (event->button == 1)
{
if (view->priv->toggle_column >= 0)
{
path_toggled (NULL, path, NULL, view);
}
else
{
view->priv->drag_started = FALSE;
view->priv->drag_button = event->button;
view->priv->drag_x = event->x;
view->priv->drag_y = event->y;
}
}
gtk_tree_path_free (path);
}
else
{
gtk_tree_selection_unselect_all (selection);
}
return TRUE;
}
static void
ephy_node_view_set_filter (EphyNodeView *view, EphyNodeFilter *filter)
{
gboolean refilter = FALSE;
if (view->priv->filter)
{
g_object_unref (view->priv->filter);
refilter = TRUE;
}
if (filter)
{
view->priv->filter = g_object_ref (filter);
g_signal_connect_object (G_OBJECT (view->priv->filter),
"changed", G_CALLBACK (filter_changed_cb),
G_OBJECT (view), 0);
}
if (refilter)
{
gtk_tree_model_filter_refilter
(GTK_TREE_MODEL_FILTER (view->priv->filtermodel));
}
}
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:
ephy_node_view_set_filter (view, g_value_get_object (value));
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;
}
}
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));
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)
{
GtkTreePath *path;
GtkTreeIter iter, iter2;
EphyNode *node;
view->priv->edited_node = NULL;
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);
gtk_tree_model_filter_convert_iter_to_child_iter
(GTK_TREE_MODEL_FILTER (view->priv->filtermodel), &iter, &iter2);
node = ephy_tree_model_node_node_from_iter (view->priv->nodemodel, &iter);
ephy_node_set_property_string (node, view->priv->editable_property,
new_text);
gtk_tree_path_free (path);
view->priv->remove_if_cancelled = FALSE;
}
static void
renderer_editing_canceled_cb (GtkCellRendererText *cell,
EphyNodeView *view)
{
if (view->priv->remove_if_cancelled)
{
ephy_node_unref (view->priv->edited_node);
view->priv->remove_if_cancelled = FALSE;
}
}
static inline int
compare_string_values (const GValue *a_value, const GValue *b_value)
{
const char *str1, *str2;
int retval;
str1 = g_value_get_string (a_value);
str2 = g_value_get_string (b_value);
if (str1 == NULL)
{
retval = -1;
}
else if (str2 == NULL)
{
retval = 1;
}
else
{
char *str_a;
char *str_b;
str_a = g_utf8_casefold (str1, -1);
str_b = g_utf8_casefold (str2, -1);
retval = g_utf8_collate (str_a, str_b);
g_free (str_a);
g_free (str_b);
}
return retval;
}
static int
ephy_node_view_sort_func (GtkTreeModel *model,
GtkTreeIter *a,
GtkTreeIter *b,
EphyNodeView *view)
{
GValue a_value = {0, };
GValue b_value = {0, };
int p_column, column, retval = 0;
GtkSortType sort_type;
g_return_val_if_fail (model != NULL, 0);
g_return_val_if_fail (view != NULL, 0);
p_column = view->priv->priority_column;
column = view->priv->sort_column;
sort_type = view->priv->sort_type;
if (p_column >= 0)
{
gtk_tree_model_get_value (model, a, p_column, &a_value);
gtk_tree_model_get_value (model, b, p_column, &b_value);
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;
}
g_value_unset (&a_value);
g_value_unset (&b_value);
}
if (retval == 0)
{
GType type;
type = gtk_tree_model_get_column_type (model, column);
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:
retval = compare_string_values (&a_value, &b_value);
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);
}
if (sort_type == GTK_SORT_DESCENDING)
{
if (retval > 0)
{
retval = -1;
}
else if (retval < 0)
{
retval = 1;
}
}
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);
}
}
int
ephy_node_view_add_data_column (EphyNodeView *view,
GType value_type,
guint prop_id,
EphyTreeModelNodeValueFunc func,
gpointer data)
{
int column;
if (func)
{
column = ephy_tree_model_node_add_func_column
(view->priv->nodemodel, value_type, func, data);
}
else
{
column = ephy_tree_model_node_add_prop_column
(view->priv->nodemodel, value_type, prop_id);
}
return column;
}
int
ephy_node_view_add_column (EphyNodeView *view,
const char *title,
GType value_type,
guint prop_id,
EphyNodeViewFlags flags,
EphyTreeModelNodeValueFunc icon_func,
GtkTreeViewColumn **ret)
{
GtkTreeViewColumn *gcolumn;
GtkCellRenderer *renderer;
int column;
int icon_column;
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);
g_signal_connect (renderer, "editing-canceled",
G_CALLBACK (renderer_editing_canceled_cb), view);
}
gtk_tree_view_column_pack_start (gcolumn, renderer, TRUE);
gtk_tree_view_column_set_attributes (gcolumn, renderer,
"text", column,
NULL);
gtk_tree_view_column_set_title (gcolumn, title);
gtk_tree_view_append_column (GTK_TREE_VIEW (view),
gcolumn);
if (flags & EPHY_NODE_VIEW_SHOW_PRIORITY)
{
int wcol;
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);
}
if (flags & EPHY_NODE_VIEW_SORTABLE)
{
/* Now we have created a new column, re-create the
* sort model, but ensure that the set_sort function
* hasn't been called, see bug #320686 */
g_assert (view->priv->sort_column == -1);
g_object_unref (view->priv->sortmodel);
view->priv->sortmodel = ephy_tree_model_sort_new (view->priv->filtermodel);
gtk_tree_view_set_model (GTK_TREE_VIEW (view), GTK_TREE_MODEL (view->priv->sortmodel));
gtk_tree_view_column_set_sort_column_id (gcolumn, column);
}
if (flags & EPHY_NODE_VIEW_SEARCHABLE)
{
gtk_tree_view_set_search_column (GTK_TREE_VIEW (view), column);
gtk_tree_view_set_enable_search (GTK_TREE_VIEW (view), TRUE);
}
if (flags & EPHY_NODE_VIEW_ELLIPSIZED)
{
g_object_set (renderer, "ellipsize-set", TRUE,
"ellipsize", PANGO_ELLIPSIZE_END, NULL);
}
if (ret != NULL)
*ret = gcolumn;
return column;
}
void
ephy_node_view_set_priority (EphyNodeView *view, guint priority_prop_id)
{
int priority_column;
priority_column = ephy_tree_model_node_add_func_column
(view->priv->nodemodel, G_TYPE_INT,
(EphyTreeModelNodeValueFunc) provide_priority,
view);
view->priv->priority_column = priority_column;
view->priv->priority_prop_id = priority_prop_id;
}
void
ephy_node_view_set_sort (EphyNodeView *view, GType value_type, guint prop_id,
GtkSortType sort_type)
{
GtkTreeSortable *sortable = GTK_TREE_SORTABLE (view->priv->sortmodel);
int column;
column = ephy_tree_model_node_add_prop_column
(view->priv->nodemodel, value_type, prop_id);
view->priv->sort_column = column;
view->priv->sort_type = sort_type;
gtk_tree_sortable_set_default_sort_func
(sortable, (GtkTreeIterCompareFunc)ephy_node_view_sort_func,
view, NULL);
gtk_tree_sortable_set_sort_column_id
(sortable, GTK_TREE_SORTABLE_DEFAULT_SORT_COLUMN_ID,
sort_type);
}
static void
ephy_node_view_init (EphyNodeView *view)
{
view->priv = EPHY_NODE_VIEW_GET_PRIVATE (view);
view->priv->toggle_column = -1;
view->priv->priority_column = -1;
view->priv->priority_prop_id = 0;
view->priv->sort_column = -1;
view->priv->sort_type = GTK_SORT_ASCENDING;
gtk_tree_view_set_enable_search (GTK_TREE_VIEW (view), FALSE);
}
static void
get_selection (GtkTreeModel *model,
GtkTreePath *path,
GtkTreeIter *iter,
gpointer *data)
{
GList **list = data[0];
EphyNodeView *view = EPHY_NODE_VIEW (data[1]);
EphyNode *node;
node = get_node_from_path (view, path);
*list = g_list_prepend (*list, node);
}
GList *
ephy_node_view_get_selection (EphyNodeView *view)
{
GList *list = NULL;
GtkTreeSelection *selection;
gpointer data[2];
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view));
data[0] = &list;
data[1] = view;
gtk_tree_selection_selected_foreach
(selection,
(GtkTreeSelectionForeachFunc) get_selection,
(gpointer) data);
return list;
}
void
ephy_node_view_remove (EphyNodeView *view)
{
EphyNodeViewPrivate *priv = view->priv;
GList *list, *l;
EphyNode *node;
GtkTreeIter iter, iter2, iter3;
GtkTreePath *path;
GtkTreeRowReference *row_ref = NULL;
GtkTreeSelection *selection;
/* 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);
if (list == NULL) return;
node = g_list_first (list)->data;
ephy_tree_model_node_iter_from_node (EPHY_TREE_MODEL_NODE (view->priv->nodemodel),
node, &iter3);
gtk_tree_model_filter_convert_child_iter_to_iter (GTK_TREE_MODEL_FILTER (view->priv->filtermodel),
&iter2, &iter3);
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);
/* Work around bug #346662 */
priv->changing_selection = TRUE;
for (l = list; l != NULL; l = l->next)
{
ephy_node_unref (l->data);
}
priv->changing_selection = FALSE;
g_list_free (list);
/* Fake a selection changed signal */
selection = gtk_tree_view_get_selection (GTK_TREE_VIEW (view));
g_signal_emit_by_name (selection, "changed");
/* Select the "next" node */
if (row_ref != NULL)
{
path = gtk_tree_row_reference_get_path (row_ref);
if (path != NULL)
{
gtk_tree_view_set_cursor (GTK_TREE_VIEW (view), path, NULL, FALSE);
gtk_tree_path_free (path);
}
gtk_tree_row_reference_free (row_ref);
}
}
void
ephy_node_view_select_node (EphyNodeView *view,
EphyNode *node)
{
GtkTreeIter iter, iter2;
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_filter_convert_child_iter_to_iter (GTK_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,
const GtkTargetEntry *types,
int n_types,
int base_drag_column,
int extra_drag_column)
{
GtkWidget *treeview;
g_return_if_fail (view != NULL);
treeview = GTK_WIDGET (view);
view->priv->source_target_list =
gtk_target_list_new (types, n_types);
ephy_tree_model_sort_set_base_drag_column_id (EPHY_TREE_MODEL_SORT (view->priv->sortmodel),
base_drag_column);
ephy_tree_model_sort_set_extra_drag_column_id (EPHY_TREE_MODEL_SORT (view->priv->sortmodel),
extra_drag_column);
g_signal_connect_object (G_OBJECT (view),
"button_release_event",
G_CALLBACK (button_release_cb),
view,
0);
g_signal_connect_object (G_OBJECT (view),
"motion_notify_event",
G_CALLBACK (motion_notify_cb),
view,
0);
g_signal_connect_object (G_OBJECT (view),
"drag_data_get",
G_CALLBACK (drag_data_get_cb),
view,
0);
}
void
ephy_node_view_edit (EphyNodeView *view, gboolean remove_if_cancelled)
{
GtkTreePath *path;
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;
path = rows->data;
g_object_set (G_OBJECT (view->priv->editable_renderer),
"editable", TRUE,
NULL);
gtk_tree_view_set_cursor (GTK_TREE_VIEW (view), path,
view->priv->editable_column,
TRUE);
view->priv->edited_node = get_node_from_path (view, path);
view->priv->remove_if_cancelled = remove_if_cancelled;
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));
}
static gboolean
filter_visible_func (GtkTreeModel *model, GtkTreeIter *iter, gpointer data)
{
EphyNode *node;
EphyNodeView *view = EPHY_NODE_VIEW (data);
if (view->priv->filter)
{
node = ephy_tree_model_node_node_from_iter (view->priv->nodemodel, iter);
return ephy_node_filter_evaluate (view->priv->filter, node);
}
return TRUE;
}
static GObject *
ephy_node_view_constructor (GType type, guint n_construct_properties,
GObjectConstructParam *construct_params)
{
GObject *object;
EphyNodeView *view;
EphyNodeViewPrivate *priv;
GtkTreeSelection *selection;
object = parent_class->constructor (type, n_construct_properties,
construct_params);
view = EPHY_NODE_VIEW (object);
priv = EPHY_NODE_VIEW_GET_PRIVATE (object);
priv->nodemodel = ephy_tree_model_node_new (priv->root);
priv->filtermodel = gtk_tree_model_filter_new (GTK_TREE_MODEL (priv->nodemodel),
NULL);
gtk_tree_model_filter_set_visible_func (GTK_TREE_MODEL_FILTER (priv->filtermodel),
filter_visible_func, view, NULL);
priv->sortmodel = ephy_tree_model_sort_new (priv->filtermodel);
gtk_tree_view_set_model (GTK_TREE_VIEW (object), GTK_TREE_MODEL (priv->sortmodel));
g_signal_connect_object (object, "button_press_event",
G_CALLBACK (ephy_node_view_button_press_cb),
view, 0);
g_signal_connect (object, "key_press_event",
G_CALLBACK (ephy_node_view_key_press_cb),
view);
g_signal_connect_object (object, "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);
return object;
}
void
ephy_node_view_add_toggle (EphyNodeView *view, EphyTreeModelNodeValueFunc value_func,
gpointer data)
{
GtkCellRenderer *renderer;
GtkTreeViewColumn *col;
int column;
column = ephy_tree_model_node_add_func_column
(view->priv->nodemodel, G_TYPE_BOOLEAN, value_func, data);
view->priv->toggle_column = column;
renderer = gtk_cell_renderer_toggle_new ();
col = gtk_tree_view_column_new_with_attributes
("", renderer, "active", column, NULL);
gtk_tree_view_append_column (GTK_TREE_VIEW (view), col);
}
void
ephy_node_view_popup (EphyNodeView *view, GtkWidget *menu)
{
GdkEvent *event;
event = gtk_get_current_event ();
if (event)
{
if (event->type == GDK_KEY_PRESS)
{
GdkEventKey *key = (GdkEventKey *) event;
gtk_menu_popup (GTK_MENU (menu), NULL, NULL,
ephy_gui_menu_position_tree_selection,
view, 0, key->time);
gtk_menu_shell_select_first (GTK_MENU_SHELL (menu), FALSE);
}
else if (event->type == GDK_BUTTON_PRESS)
{
GdkEventButton *button = (GdkEventButton *) event;
gtk_menu_popup (GTK_MENU (menu), NULL, NULL, NULL,
NULL, button->button, button->time);
}
gdk_event_free (event);
}
}
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->constructor = ephy_node_view_constructor;
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_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB | 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_STATIC_NAME | G_PARAM_STATIC_NICK | G_PARAM_STATIC_BLURB));
ephy_node_view_signals[NODE_TOGGLED] =
g_signal_new ("node_toggled",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (EphyNodeViewClass, node_toggled),
NULL, NULL,
ephy_marshal_VOID__POINTER_BOOLEAN,
G_TYPE_NONE,
2,
G_TYPE_POINTER,
G_TYPE_BOOLEAN);
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[NODE_MIDDLE_CLICKED] =
g_signal_new ("node_middle_clicked",
G_OBJECT_CLASS_TYPE (object_class),
G_SIGNAL_RUN_LAST,
G_STRUCT_OFFSET (EphyNodeViewClass, node_middle_clicked),
NULL, NULL,
g_cclosure_marshal_VOID__POINTER,
G_TYPE_NONE,
1,
G_TYPE_POINTER);
g_type_class_add_private (object_class, sizeof (EphyNodeViewPrivate));
}