4013 lines
118 KiB
C++
4013 lines
118 KiB
C++
/*
|
|
* mooiconview.c
|
|
*
|
|
* Copyright (C) 2004-2010 by Yevgen Muntyan <emuntyan@users.sourceforge.net>
|
|
*
|
|
* This file is part of medit. medit is free software; you can
|
|
* redistribute it and/or modify it under the terms of the
|
|
* GNU Lesser General Public License as published by the
|
|
* Free Software Foundation; either version 2.1 of the License,
|
|
* or (at your option) any later version.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
* License along with medit. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
#include "mooiconview.h"
|
|
#include "marshals.h"
|
|
#include "mooutils/mooaccel.h"
|
|
#include "mooutils/mooutils-gobject.h"
|
|
#include "mooutils/mooutils-misc.h"
|
|
#include "mooutils/moocompat.h"
|
|
#include "moocpp/gobjtypes.h"
|
|
#include <gdk/gdkkeysyms.h>
|
|
#include <gtk/gtk.h>
|
|
#include <string.h>
|
|
#include <math.h>
|
|
|
|
|
|
typedef struct Column Column;
|
|
typedef struct Layout Layout;
|
|
typedef struct CellInfo CellInfo;
|
|
typedef struct Selection Selection;
|
|
typedef struct DndInfo DndInfo;
|
|
|
|
struct Column {
|
|
GtkTreePath *first;
|
|
int index;
|
|
int width;
|
|
int offset;
|
|
GPtrArray *entries;
|
|
};
|
|
|
|
|
|
struct Layout {
|
|
int num_rows;
|
|
|
|
int width;
|
|
int height;
|
|
int row_height;
|
|
|
|
int pixbuf_width;
|
|
int pixbuf_height;
|
|
int text_height;
|
|
|
|
GSList *columns;
|
|
};
|
|
|
|
|
|
struct CellInfo {
|
|
GtkCellRenderer *cell;
|
|
GSList *attributes;
|
|
MooIconCellDataFunc func;
|
|
gpointer func_data;
|
|
GDestroyNotify destroy;
|
|
gboolean show;
|
|
};
|
|
|
|
|
|
struct DndInfo {
|
|
GtkTargetList *dest_targets;
|
|
GdkDragAction dest_actions;
|
|
|
|
GdkModifierType start_button_mask;
|
|
GtkTargetList *source_targets;
|
|
GdkDragAction source_actions;
|
|
|
|
GdkDragContext *drag_motion_context;
|
|
|
|
guint drag_dest_inside : 1;
|
|
guint source_enabled : 1;
|
|
guint dest_enabled : 1;
|
|
};
|
|
|
|
|
|
struct _MooIconViewPrivate {
|
|
GtkTreeModel *model;
|
|
|
|
CellInfo pixbuf;
|
|
CellInfo text;
|
|
int pixel_icon_size;
|
|
int icon_size; /* GtkIconSize */
|
|
|
|
int xoffset;
|
|
GtkAdjustment *adjustment;
|
|
GtkTreeRowReference *scroll_to;
|
|
|
|
guint update_idle;
|
|
|
|
Layout *layout;
|
|
Selection *selection;
|
|
GtkTreeRowReference *cursor;
|
|
GtkTreeRowReference *drop_dest;
|
|
|
|
gboolean mapped;
|
|
|
|
DndInfo *dnd_info;
|
|
int button_pressed;
|
|
int button_press_x;
|
|
int button_press_y;
|
|
GtkTreeRowReference *button_press_row;
|
|
GdkModifierType button_press_mods;
|
|
|
|
guint drag_scroll_timeout;
|
|
|
|
int drag_select_x;
|
|
int drag_select_y;
|
|
gboolean drag_select;
|
|
GTree *old_selection;
|
|
GdkGC *sel_gc;
|
|
};
|
|
|
|
|
|
static void moo_icon_view_dispose (GObject *object);
|
|
static void moo_icon_view_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec);
|
|
static void moo_icon_view_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec);
|
|
|
|
static void moo_icon_view_state_changed (GtkWidget *widget,
|
|
GtkStateType previous);
|
|
static void moo_icon_view_style_set (GtkWidget *widget,
|
|
GtkStyle *previous);
|
|
static void moo_icon_view_map (GtkWidget *widget);
|
|
static void moo_icon_view_realize (GtkWidget *widget);
|
|
static void moo_icon_view_unrealize (GtkWidget *widget);
|
|
static void moo_icon_view_size_request (GtkWidget *widget,
|
|
GtkRequisition *requisition);
|
|
static void moo_icon_view_size_allocate (GtkWidget *widget,
|
|
GtkAllocation *allocation);
|
|
static gboolean moo_icon_view_expose (GtkWidget *widget,
|
|
GdkEventExpose *event);
|
|
|
|
static gboolean moo_icon_view_button_press (GtkWidget *widget,
|
|
GdkEventButton *event);
|
|
static gboolean moo_icon_view_button_release(GtkWidget *widget,
|
|
GdkEventButton *event);
|
|
static gboolean moo_icon_view_motion_notify (GtkWidget *widget,
|
|
GdkEventMotion *event);
|
|
static gboolean moo_icon_view_scroll_event (GtkWidget *widget,
|
|
GdkEventScroll *event);
|
|
|
|
static gboolean moo_icon_view_maybe_drag_select (MooIconView *view,
|
|
GdkEventMotion *event);
|
|
|
|
static gboolean moo_icon_view_maybe_drag (MooIconView *view,
|
|
GdkEventMotion *event);
|
|
static void moo_icon_view_drag_begin (GtkWidget *widget,
|
|
GdkDragContext *context);
|
|
static void moo_icon_view_drag_end (GtkWidget *widget,
|
|
GdkDragContext *context);
|
|
static void moo_icon_view_drag_leave (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
guint time);
|
|
static gboolean moo_icon_view_drag_motion (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
int x,
|
|
int y,
|
|
guint time);
|
|
static void drag_scroll_check (MooIconView *view,
|
|
int x,
|
|
int y);
|
|
static void drag_scroll_stop (MooIconView *view);
|
|
static void drag_select_finish (MooIconView *view);
|
|
|
|
static void row_changed (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
GtkTreeIter *iter,
|
|
MooIconView *view);
|
|
static void row_deleted (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
MooIconView *view);
|
|
static void row_inserted (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
GtkTreeIter *iter,
|
|
MooIconView *view);
|
|
static void rows_reordered (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
GtkTreeIter *iter,
|
|
gpointer whatsthis,
|
|
MooIconView *view);
|
|
|
|
static void cell_data_func (MooIconView *view,
|
|
GtkCellRenderer *cell,
|
|
GtkTreeModel *model,
|
|
GtkTreeIter *iter,
|
|
gpointer cell_info);
|
|
static void _moo_icon_view_set_cell (MooIconView *view,
|
|
MooIconViewCell cell_type,
|
|
GtkCellRenderer *cell);
|
|
|
|
static void _moo_icon_view_set_adjustment (MooIconView *view,
|
|
GtkAdjustment *adjustment);
|
|
static void moo_icon_view_set_scroll_adjustments
|
|
(GtkWidget *widget,
|
|
GtkAdjustment *hadj,
|
|
GtkAdjustment *vadj);
|
|
static void moo_icon_view_update_adjustment (MooIconView *view);
|
|
static void moo_icon_view_scroll_to (MooIconView *view,
|
|
int offset);
|
|
static int clamp_offset (MooIconView *view,
|
|
int offset);
|
|
|
|
static void moo_icon_view_invalidate_layout (MooIconView *view);
|
|
static gboolean moo_icon_view_update_layout (MooIconView *view);
|
|
|
|
static GtkTreePath *_moo_icon_view_get_cursor (MooIconView *view);
|
|
static void moo_icon_view_select_path (MooIconView *view,
|
|
GtkTreePath *path);
|
|
static void moo_icon_view_select_paths (MooIconView *view,
|
|
GList *list);
|
|
static void moo_icon_view_select_range (MooIconView *view,
|
|
GtkTreePath *start,
|
|
GtkTreePath *end);
|
|
static void moo_icon_view_unselect_path (MooIconView *view,
|
|
GtkTreePath *path);
|
|
static GList *moo_icon_view_get_paths_in_rect (MooIconView *view,
|
|
GdkRectangle *rect);
|
|
|
|
static void init_selection (MooIconView *view);
|
|
static void free_selection (MooIconView *view);
|
|
static void selection_clear (MooIconView *view);
|
|
static void selection_row_deleted (MooIconView *view);
|
|
static void cursor_row_deleted (MooIconView *view);
|
|
|
|
static void free_attributes (CellInfo *info);
|
|
static gboolean check_empty (MooIconView *view);
|
|
static void init_layout (MooIconView *view);
|
|
static void destroy_layout (MooIconView *view);
|
|
|
|
static void activate_item_at_cursor (MooIconView *view);
|
|
static void add_move_binding (GtkBindingSet *binding_set,
|
|
guint keyval,
|
|
guint modmask,
|
|
GtkMovementStep step,
|
|
gint count);
|
|
static gboolean move_cursor (MooIconView *view,
|
|
GtkMovementStep step,
|
|
gint count,
|
|
gboolean extend_selection);
|
|
static GtkTreePath *ensure_cursor (MooIconView *view);
|
|
|
|
static DndInfo *dnd_info_new (void);
|
|
static void dnd_info_free (DndInfo *info);
|
|
|
|
|
|
/* MOO_TYPE_ICON_VIEW */
|
|
G_DEFINE_TYPE (MooIconView, _moo_icon_view, GTK_TYPE_WIDGET)
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_PIXBUF_CELL,
|
|
PROP_TEXT_CELL,
|
|
PROP_MODEL
|
|
};
|
|
|
|
enum {
|
|
ROW_ACTIVATED,
|
|
SET_SCROLL_ADJUSTMENTS,
|
|
SELECTION_CHANGED,
|
|
CURSOR_MOVED,
|
|
|
|
ACTIVATE_ITEM_AT_CURSOR,
|
|
MOVE_CURSOR,
|
|
SELECT_ALL,
|
|
|
|
LAST_SIGNAL
|
|
};
|
|
|
|
static guint signals[LAST_SIGNAL];
|
|
|
|
static void
|
|
_moo_icon_view_class_init (MooIconViewClass *klass)
|
|
{
|
|
GtkBindingSet *binding_set;
|
|
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
|
|
GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
|
|
|
|
g_type_class_add_private (klass, sizeof (MooIconViewPrivate));
|
|
|
|
gobject_class->dispose = moo_icon_view_dispose;
|
|
gobject_class->set_property = moo_icon_view_set_property;
|
|
gobject_class->get_property = moo_icon_view_get_property;
|
|
|
|
widget_class->state_changed = moo_icon_view_state_changed;
|
|
widget_class->style_set = moo_icon_view_style_set;
|
|
widget_class->map = moo_icon_view_map;
|
|
widget_class->realize = moo_icon_view_realize;
|
|
widget_class->unrealize = moo_icon_view_unrealize;
|
|
widget_class->size_request = moo_icon_view_size_request;
|
|
widget_class->size_allocate = moo_icon_view_size_allocate;
|
|
widget_class->expose_event = moo_icon_view_expose;
|
|
widget_class->scroll_event = moo_icon_view_scroll_event;
|
|
widget_class->button_press_event = moo_icon_view_button_press;
|
|
widget_class->button_release_event = moo_icon_view_button_release;
|
|
widget_class->motion_notify_event = moo_icon_view_motion_notify;
|
|
widget_class->drag_begin = moo_icon_view_drag_begin;
|
|
widget_class->drag_end = moo_icon_view_drag_end;
|
|
widget_class->drag_leave = moo_icon_view_drag_leave;
|
|
widget_class->drag_motion = moo_icon_view_drag_motion;
|
|
|
|
klass->set_scroll_adjustments = moo_icon_view_set_scroll_adjustments;
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_PIXBUF_CELL,
|
|
g_param_spec_object ("pixbuf-cell",
|
|
"pixbuf-cell",
|
|
"pixbuf-cell",
|
|
GTK_TYPE_CELL_RENDERER,
|
|
(GParamFlags) G_PARAM_READWRITE));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_TEXT_CELL,
|
|
g_param_spec_object ("text-cell",
|
|
"text-cell",
|
|
"text-cell",
|
|
GTK_TYPE_CELL_RENDERER,
|
|
(GParamFlags) G_PARAM_READWRITE));
|
|
|
|
g_object_class_install_property (gobject_class,
|
|
PROP_MODEL,
|
|
g_param_spec_object ("model",
|
|
"model",
|
|
"model",
|
|
GTK_TYPE_TREE_MODEL,
|
|
(GParamFlags) G_PARAM_READWRITE));
|
|
|
|
signals[ROW_ACTIVATED] =
|
|
g_signal_new ("row-activated",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (MooIconViewClass, row_activated),
|
|
NULL, NULL,
|
|
_moo_marshal_VOID__BOXED,
|
|
G_TYPE_NONE, 1,
|
|
GTK_TYPE_TREE_PATH | G_SIGNAL_TYPE_STATIC_SCOPE);
|
|
|
|
signals[SELECTION_CHANGED] =
|
|
g_signal_new ("selection-changed",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (MooIconViewClass, selection_changed),
|
|
NULL, NULL,
|
|
_moo_marshal_VOID__VOID,
|
|
G_TYPE_NONE, 0);
|
|
|
|
signals[CURSOR_MOVED] =
|
|
g_signal_new ("cursor-moved",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (MooIconViewClass, cursor_moved),
|
|
NULL, NULL,
|
|
_moo_marshal_VOID__VOID,
|
|
G_TYPE_NONE, 1,
|
|
GTK_TYPE_TREE_PATH | G_SIGNAL_TYPE_STATIC_SCOPE);
|
|
|
|
signals[SET_SCROLL_ADJUSTMENTS] =
|
|
g_signal_new ("set-scroll-adjustments",
|
|
G_OBJECT_CLASS_TYPE (gobject_class),
|
|
G_SIGNAL_RUN_LAST,
|
|
G_STRUCT_OFFSET (MooIconViewClass, set_scroll_adjustments),
|
|
NULL, NULL,
|
|
_moo_marshal_VOID__OBJECT_OBJECT,
|
|
G_TYPE_NONE, 2,
|
|
GTK_TYPE_ADJUSTMENT,
|
|
GTK_TYPE_ADJUSTMENT);
|
|
widget_class->set_scroll_adjustments_signal = signals[SET_SCROLL_ADJUSTMENTS];
|
|
|
|
signals[ACTIVATE_ITEM_AT_CURSOR] =
|
|
_moo_signal_new_cb ("activate-item-at-cursor",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_CALLBACK (activate_item_at_cursor),
|
|
NULL, NULL,
|
|
_moo_marshal_VOID__VOID,
|
|
G_TYPE_NONE, 0);
|
|
|
|
signals[MOVE_CURSOR] =
|
|
_moo_signal_new_cb ("move-cursor",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_CALLBACK (move_cursor),
|
|
NULL, NULL,
|
|
_moo_marshal_BOOLEAN__ENUM_INT_BOOLEAN,
|
|
G_TYPE_BOOLEAN, 3,
|
|
GTK_TYPE_MOVEMENT_STEP,
|
|
G_TYPE_INT, G_TYPE_BOOLEAN);
|
|
|
|
signals[SELECT_ALL] =
|
|
_moo_signal_new_cb ("select-all",
|
|
G_OBJECT_CLASS_TYPE (klass),
|
|
G_SIGNAL_RUN_LAST | G_SIGNAL_ACTION,
|
|
G_CALLBACK (_moo_icon_view_select_all),
|
|
NULL, NULL,
|
|
_moo_marshal_VOID__VOID,
|
|
G_TYPE_NONE, 0);
|
|
|
|
binding_set = gtk_binding_set_by_class (klass);
|
|
|
|
gtk_binding_entry_add_signal (binding_set, GDK_Return, GdkModifierType (0), "activate-item-at-cursor", 0);
|
|
gtk_binding_entry_add_signal (binding_set, GDK_ISO_Enter, GdkModifierType (0), "activate-item-at-cursor", 0);
|
|
gtk_binding_entry_add_signal (binding_set, GDK_KP_Enter, GdkModifierType (0), "activate-item-at-cursor", 0);
|
|
gtk_binding_entry_add_signal (binding_set, GDK_a, MOO_ACCEL_CTRL_MASK, "select-all", 0);
|
|
|
|
add_move_binding (binding_set, GDK_Up, 0,
|
|
GTK_MOVEMENT_DISPLAY_LINES, -1);
|
|
add_move_binding (binding_set, GDK_KP_Up, 0,
|
|
GTK_MOVEMENT_DISPLAY_LINES, -1);
|
|
|
|
add_move_binding (binding_set, GDK_Down, 0,
|
|
GTK_MOVEMENT_DISPLAY_LINES, 1);
|
|
add_move_binding (binding_set, GDK_KP_Down, 0,
|
|
GTK_MOVEMENT_DISPLAY_LINES, 1);
|
|
|
|
add_move_binding (binding_set, GDK_Home, 0,
|
|
GTK_MOVEMENT_BUFFER_ENDS, -1);
|
|
add_move_binding (binding_set, GDK_KP_Home, 0,
|
|
GTK_MOVEMENT_BUFFER_ENDS, -1);
|
|
|
|
add_move_binding (binding_set, GDK_End, 0,
|
|
GTK_MOVEMENT_BUFFER_ENDS, 1);
|
|
add_move_binding (binding_set, GDK_KP_End, 0,
|
|
GTK_MOVEMENT_BUFFER_ENDS, 1);
|
|
|
|
add_move_binding (binding_set, GDK_Page_Up, 0,
|
|
GTK_MOVEMENT_PAGES, -1);
|
|
add_move_binding (binding_set, GDK_KP_Page_Up, 0,
|
|
GTK_MOVEMENT_PAGES, -1);
|
|
|
|
add_move_binding (binding_set, GDK_Page_Down, 0,
|
|
GTK_MOVEMENT_PAGES, 1);
|
|
add_move_binding (binding_set, GDK_KP_Page_Down, 0,
|
|
GTK_MOVEMENT_PAGES, 1);
|
|
|
|
add_move_binding (binding_set, GDK_Right, 0,
|
|
GTK_MOVEMENT_VISUAL_POSITIONS, 1);
|
|
add_move_binding (binding_set, GDK_Left, 0,
|
|
GTK_MOVEMENT_VISUAL_POSITIONS, -1);
|
|
|
|
add_move_binding (binding_set, GDK_KP_Right, 0,
|
|
GTK_MOVEMENT_VISUAL_POSITIONS, 1);
|
|
add_move_binding (binding_set, GDK_KP_Left, 0,
|
|
GTK_MOVEMENT_VISUAL_POSITIONS, -1);
|
|
}
|
|
|
|
|
|
static void add_move_binding (GtkBindingSet *binding_set,
|
|
guint keyval,
|
|
guint modmask,
|
|
GtkMovementStep step,
|
|
gint count)
|
|
{
|
|
gtk_binding_entry_add_signal (binding_set, keyval,
|
|
GdkModifierType (modmask),
|
|
"move_cursor", 3,
|
|
G_TYPE_ENUM, step,
|
|
G_TYPE_INT, count,
|
|
G_TYPE_BOOLEAN, FALSE);
|
|
|
|
gtk_binding_entry_add_signal (binding_set, keyval,
|
|
GdkModifierType(modmask | GDK_SHIFT_MASK),
|
|
"move_cursor", 3,
|
|
G_TYPE_ENUM, step,
|
|
G_TYPE_INT, count,
|
|
G_TYPE_BOOLEAN, TRUE);
|
|
|
|
if (modmask & GDK_CONTROL_MASK)
|
|
return;
|
|
|
|
gtk_binding_entry_add_signal (binding_set, keyval,
|
|
GDK_CONTROL_MASK,
|
|
"move_cursor", 3,
|
|
G_TYPE_ENUM, step,
|
|
G_TYPE_INT, count,
|
|
G_TYPE_BOOLEAN, FALSE);
|
|
|
|
gtk_binding_entry_add_signal (binding_set, keyval,
|
|
GDK_CONTROL_MASK | GDK_SHIFT_MASK,
|
|
"move_cursor", 3,
|
|
G_TYPE_ENUM, step,
|
|
G_TYPE_INT, count,
|
|
G_TYPE_BOOLEAN, TRUE);
|
|
}
|
|
|
|
|
|
static void
|
|
_moo_icon_view_init (MooIconView *view)
|
|
{
|
|
GtkWidget *widget = GTK_WIDGET (view);
|
|
|
|
widget->allocation.width = -1;
|
|
widget->allocation.height = -1;
|
|
|
|
GTK_WIDGET_UNSET_NO_WINDOW (view);
|
|
GTK_WIDGET_SET_CAN_FOCUS (view);
|
|
|
|
view->priv = G_TYPE_INSTANCE_GET_PRIVATE (view, MOO_TYPE_ICON_VIEW, MooIconViewPrivate);
|
|
|
|
view->priv->pixbuf.cell = gtk_cell_renderer_pixbuf_new ();
|
|
g_object_ref_sink (view->priv->pixbuf.cell);
|
|
view->priv->pixbuf.attributes = NULL;
|
|
view->priv->pixbuf.func = cell_data_func;
|
|
view->priv->pixbuf.func_data = &view->priv->pixbuf;
|
|
view->priv->pixbuf.destroy = NULL;
|
|
view->priv->pixbuf.show = TRUE;
|
|
|
|
view->priv->text.cell = gtk_cell_renderer_text_new ();
|
|
g_object_ref_sink (view->priv->text.cell);
|
|
view->priv->text.attributes = NULL;
|
|
view->priv->text.func = cell_data_func;
|
|
view->priv->text.func_data = &view->priv->text;
|
|
view->priv->text.destroy = NULL;
|
|
view->priv->text.show = TRUE;
|
|
|
|
view->priv->pixel_icon_size = -1;
|
|
view->priv->icon_size = -1;
|
|
|
|
view->priv->xoffset = 0;
|
|
_moo_icon_view_set_adjustment (view, NULL);
|
|
|
|
view->priv->update_idle = 0;
|
|
|
|
init_layout (view);
|
|
init_selection (view);
|
|
|
|
view->priv->dnd_info = dnd_info_new ();
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_dispose (GObject *object)
|
|
{
|
|
MooIconView *view = MOO_ICON_VIEW (object);
|
|
|
|
_moo_icon_view_set_model (view, NULL);
|
|
|
|
g_object_unref (view->priv->adjustment);
|
|
view->priv->adjustment = NULL;
|
|
|
|
if (view->priv->update_idle)
|
|
{
|
|
g_source_remove (view->priv->update_idle);
|
|
view->priv->update_idle = 0;
|
|
}
|
|
|
|
if (view->priv->pixbuf.cell)
|
|
{
|
|
g_object_unref (view->priv->pixbuf.cell);
|
|
free_attributes (&view->priv->pixbuf);
|
|
if (view->priv->pixbuf.destroy)
|
|
view->priv->pixbuf.destroy (view->priv->pixbuf.func_data);
|
|
view->priv->pixbuf.cell = NULL;
|
|
}
|
|
|
|
if (view->priv->text.cell)
|
|
{
|
|
g_object_unref (view->priv->text.cell);
|
|
free_attributes (&view->priv->text);
|
|
if (view->priv->text.destroy)
|
|
view->priv->text.destroy (view->priv->text.func_data);
|
|
view->priv->text.cell = NULL;
|
|
}
|
|
|
|
destroy_layout (view);
|
|
free_selection (view);
|
|
|
|
dnd_info_free (view->priv->dnd_info);
|
|
view->priv->dnd_info = NULL;
|
|
|
|
G_OBJECT_CLASS (_moo_icon_view_parent_class)->dispose (object);
|
|
}
|
|
|
|
|
|
static DndInfo *
|
|
dnd_info_new (void)
|
|
{
|
|
return g_slice_new0 (DndInfo);
|
|
}
|
|
|
|
static void
|
|
dnd_info_free (DndInfo *info)
|
|
{
|
|
if (info)
|
|
{
|
|
if (info->dest_targets)
|
|
gtk_target_list_unref (info->dest_targets);
|
|
if (info->source_targets)
|
|
gtk_target_list_unref (info->source_targets);
|
|
if (info->drag_motion_context)
|
|
g_object_unref (info->drag_motion_context);
|
|
g_slice_free (DndInfo, info);
|
|
}
|
|
}
|
|
|
|
|
|
GtkWidget *
|
|
_moo_icon_view_new (GtkTreeModel *model)
|
|
{
|
|
return GTK_WIDGET (g_object_new (MOO_TYPE_ICON_VIEW,
|
|
"model", model,
|
|
(const char*) NULL));
|
|
}
|
|
|
|
|
|
static void moo_icon_view_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
MooIconView *view = MOO_ICON_VIEW (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_MODEL:
|
|
_moo_icon_view_set_model (view, reinterpret_cast<GtkTreeModel*> (g_value_get_object (value)));
|
|
break;
|
|
case PROP_PIXBUF_CELL:
|
|
_moo_icon_view_set_cell (view,
|
|
MOO_ICON_VIEW_CELL_PIXBUF,
|
|
GTK_CELL_RENDERER (g_value_get_object (value)));
|
|
break;
|
|
case PROP_TEXT_CELL:
|
|
_moo_icon_view_set_cell (view,
|
|
MOO_ICON_VIEW_CELL_TEXT,
|
|
GTK_CELL_RENDERER (g_value_get_object (value)));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
}
|
|
}
|
|
|
|
|
|
static void moo_icon_view_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
MooIconView *view = MOO_ICON_VIEW (object);
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_MODEL:
|
|
g_value_set_object (value, view->priv->model);
|
|
break;
|
|
case PROP_PIXBUF_CELL:
|
|
g_value_set_object (value, view->priv->pixbuf.cell);
|
|
break;
|
|
case PROP_TEXT_CELL:
|
|
g_value_set_object (value, view->priv->text.cell);
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
}
|
|
}
|
|
|
|
|
|
GtkTreeModel *
|
|
_moo_icon_view_get_model (MooIconView *view)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_ICON_VIEW (view), NULL);
|
|
return view->priv->model;
|
|
}
|
|
|
|
|
|
GtkCellRenderer *
|
|
_moo_icon_view_get_cell (MooIconView *view,
|
|
MooIconViewCell cell_type)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_ICON_VIEW (view), NULL);
|
|
g_return_val_if_fail (cell_type == MOO_ICON_VIEW_CELL_PIXBUF ||
|
|
cell_type == MOO_ICON_VIEW_CELL_TEXT, NULL);
|
|
if (cell_type == MOO_ICON_VIEW_CELL_PIXBUF)
|
|
return view->priv->pixbuf.cell;
|
|
else
|
|
return view->priv->text.cell;
|
|
}
|
|
|
|
|
|
void
|
|
_moo_icon_view_set_model (MooIconView *view,
|
|
GtkTreeModel *model)
|
|
{
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (!model || GTK_IS_TREE_MODEL (model));
|
|
|
|
if (view->priv->model == model)
|
|
return;
|
|
|
|
if (view->priv->model)
|
|
{
|
|
g_signal_handlers_disconnect_by_func (view->priv->model,
|
|
(gpointer) row_changed,
|
|
view);
|
|
g_signal_handlers_disconnect_by_func (view->priv->model,
|
|
(gpointer) row_deleted,
|
|
view);
|
|
g_signal_handlers_disconnect_by_func (view->priv->model,
|
|
(gpointer) row_inserted,
|
|
view);
|
|
g_signal_handlers_disconnect_by_func (view->priv->model,
|
|
(gpointer) rows_reordered,
|
|
view);
|
|
|
|
g_object_unref (view->priv->model);
|
|
view->priv->model = NULL;
|
|
|
|
selection_clear (view);
|
|
|
|
if (view->priv->button_press_row)
|
|
gtk_tree_row_reference_free (view->priv->button_press_row);
|
|
view->priv->button_press_row = NULL;
|
|
|
|
gtk_tree_row_reference_free (view->priv->cursor);
|
|
gtk_tree_row_reference_free (view->priv->drop_dest);
|
|
gtk_tree_row_reference_free (view->priv->scroll_to);
|
|
view->priv->cursor = NULL;
|
|
view->priv->drop_dest = NULL;
|
|
view->priv->scroll_to = NULL;
|
|
}
|
|
|
|
if (model)
|
|
{
|
|
view->priv->model = model;
|
|
g_object_ref (model);
|
|
|
|
g_signal_connect (model, "row-changed",
|
|
G_CALLBACK (row_changed), view);
|
|
g_signal_connect (model, "row-deleted",
|
|
G_CALLBACK (row_deleted), view);
|
|
g_signal_connect (model, "row-inserted",
|
|
G_CALLBACK (row_inserted), view);
|
|
g_signal_connect (model, "rows-reordered",
|
|
G_CALLBACK (rows_reordered), view);
|
|
}
|
|
|
|
moo_icon_view_invalidate_layout (view);
|
|
|
|
g_object_notify (G_OBJECT (view), "model");
|
|
}
|
|
|
|
|
|
static void
|
|
_moo_icon_view_set_cell (MooIconView *view,
|
|
MooIconViewCell cell_type,
|
|
GtkCellRenderer*cell)
|
|
{
|
|
CellInfo *info;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (GTK_IS_CELL_RENDERER (cell));
|
|
g_return_if_fail (cell_type == MOO_ICON_VIEW_CELL_PIXBUF ||
|
|
cell_type == MOO_ICON_VIEW_CELL_TEXT);
|
|
|
|
if (cell_type == MOO_ICON_VIEW_CELL_PIXBUF)
|
|
info = &view->priv->pixbuf;
|
|
else
|
|
info = &view->priv->text;
|
|
|
|
if (info->cell == cell)
|
|
return;
|
|
|
|
g_object_unref (info->cell);
|
|
|
|
info->cell = cell;
|
|
g_object_ref_sink (cell);
|
|
|
|
moo_icon_view_invalidate_layout (view);
|
|
}
|
|
|
|
|
|
static void free_attributes (CellInfo *info)
|
|
{
|
|
GSList *l;
|
|
for (l = info->attributes; l != NULL; l = l->next->next)
|
|
g_free (l->data);
|
|
g_slist_free (info->attributes);
|
|
info->attributes = NULL;
|
|
}
|
|
|
|
|
|
#if 0
|
|
static void
|
|
_moo_icon_view_clear_attributes (MooIconView *view,
|
|
MooIconViewCell cell)
|
|
{
|
|
CellInfo *info;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (cell == MOO_ICON_VIEW_CELL_PIXBUF ||
|
|
cell == MOO_ICON_VIEW_CELL_TEXT);
|
|
|
|
if (cell == MOO_ICON_VIEW_CELL_PIXBUF)
|
|
info = &view->priv->pixbuf;
|
|
else
|
|
info = &view->priv->text;
|
|
|
|
free_attributes (info);
|
|
|
|
moo_icon_view_invalidate_layout (view);
|
|
}
|
|
|
|
static void
|
|
moo_icon_view_set_attributesv (MooIconView *view,
|
|
MooIconViewCell cell_type,
|
|
const char *first_attr,
|
|
va_list args)
|
|
{
|
|
char *attribute;
|
|
int column;
|
|
CellInfo *info;
|
|
|
|
_moo_icon_view_clear_attributes (view, cell_type);
|
|
|
|
attribute = (char*) first_attr;
|
|
|
|
if (cell_type == MOO_ICON_VIEW_CELL_PIXBUF)
|
|
info = &view->priv->pixbuf;
|
|
else
|
|
info = &view->priv->text;
|
|
|
|
while (attribute != NULL)
|
|
{
|
|
column = va_arg (args, int);
|
|
|
|
info->attributes = g_slist_prepend (info->attributes,
|
|
GINT_TO_POINTER (column));
|
|
info->attributes = g_slist_prepend (info->attributes,
|
|
g_strdup (attribute));
|
|
|
|
attribute = va_arg (args, char*);
|
|
}
|
|
|
|
moo_icon_view_invalidate_layout (view);
|
|
}
|
|
|
|
void
|
|
_moo_icon_view_set_attributes (MooIconView *view,
|
|
MooIconViewCell cell_type,
|
|
const char *first_attr,
|
|
...)
|
|
{
|
|
va_list args;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (cell_type == MOO_ICON_VIEW_CELL_PIXBUF ||
|
|
cell_type == MOO_ICON_VIEW_CELL_TEXT);
|
|
|
|
va_start (args, first_attr);
|
|
moo_icon_view_set_attributesv (view, cell_type, first_attr, args);
|
|
va_end (args);
|
|
}
|
|
#endif
|
|
|
|
|
|
void
|
|
_moo_icon_view_set_cell_data_func (MooIconView *view,
|
|
MooIconViewCell cell,
|
|
MooIconCellDataFunc func,
|
|
gpointer func_data,
|
|
GDestroyNotify destroy)
|
|
{
|
|
CellInfo *info;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (cell == MOO_ICON_VIEW_CELL_PIXBUF ||
|
|
cell == MOO_ICON_VIEW_CELL_TEXT);
|
|
|
|
if (cell == MOO_ICON_VIEW_CELL_PIXBUF)
|
|
info = &view->priv->pixbuf;
|
|
else
|
|
info = &view->priv->text;
|
|
|
|
if (info->destroy)
|
|
info->destroy (info->func_data);
|
|
|
|
if (!func)
|
|
{
|
|
info->func = cell_data_func;
|
|
info->func_data = info;
|
|
info->destroy = NULL;
|
|
}
|
|
else
|
|
{
|
|
info->func = func;
|
|
info->func_data = func_data;
|
|
info->destroy = destroy;
|
|
}
|
|
|
|
moo_icon_view_invalidate_layout (view);
|
|
}
|
|
|
|
|
|
static gboolean
|
|
check_empty (MooIconView *view)
|
|
{
|
|
return !view->priv->model || !view->priv->layout->columns;
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_map (GtkWidget *widget)
|
|
{
|
|
GTK_WIDGET_CLASS(_moo_icon_view_parent_class)->map (widget);
|
|
moo_icon_view_invalidate_layout (MOO_ICON_VIEW (widget));
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_style_set (GtkWidget *widget,
|
|
G_GNUC_UNUSED GtkStyle *previous_style)
|
|
{
|
|
MooIconView *view = MOO_ICON_VIEW (widget);
|
|
|
|
if (GTK_WIDGET_REALIZED (widget))
|
|
gdk_window_set_background (widget->window,
|
|
&widget->style->base[GTK_WIDGET_STATE (widget)]);
|
|
|
|
if (view->priv->sel_gc)
|
|
{
|
|
g_object_unref (view->priv->sel_gc);
|
|
view->priv->sel_gc = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_state_changed (GtkWidget *widget,
|
|
G_GNUC_UNUSED GtkStateType previous_state)
|
|
{
|
|
if (GTK_WIDGET_REALIZED (widget))
|
|
gdk_window_set_background (widget->window,
|
|
&widget->style->base[GTK_WIDGET_STATE (widget)]);
|
|
|
|
if (!GTK_WIDGET_IS_SENSITIVE (widget))
|
|
_moo_icon_view_unselect_all (MOO_ICON_VIEW (widget));
|
|
|
|
gtk_widget_queue_draw (widget);
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_realize (GtkWidget *widget)
|
|
{
|
|
static GdkWindowAttr attributes;
|
|
gint attributes_mask;
|
|
MooIconView *view;
|
|
|
|
view = MOO_ICON_VIEW (widget);
|
|
|
|
GTK_WIDGET_SET_REALIZED (widget);
|
|
|
|
attributes.x = widget->allocation.x;
|
|
attributes.y = widget->allocation.y;
|
|
attributes.width = widget->allocation.width;
|
|
attributes.height = widget->allocation.height;
|
|
attributes.window_type = GDK_WINDOW_CHILD;
|
|
attributes.event_mask = gtk_widget_get_events (widget)
|
|
| GDK_POINTER_MOTION_MASK
|
|
| GDK_BUTTON_PRESS_MASK
|
|
| GDK_BUTTON_RELEASE_MASK
|
|
| GDK_EXPOSURE_MASK
|
|
| GDK_ENTER_NOTIFY_MASK
|
|
| GDK_LEAVE_NOTIFY_MASK;
|
|
|
|
attributes.visual = gtk_widget_get_visual (widget);
|
|
attributes.colormap = gtk_widget_get_colormap (widget);
|
|
attributes.wclass = GDK_INPUT_OUTPUT;
|
|
|
|
attributes_mask = GDK_WA_X | GDK_WA_Y | GDK_WA_VISUAL | GDK_WA_COLORMAP;
|
|
|
|
widget->window = gdk_window_new (gtk_widget_get_parent_window (widget),
|
|
&attributes, attributes_mask);
|
|
gdk_window_set_user_data (widget->window, widget);
|
|
|
|
widget->style = gtk_style_attach (widget->style, widget->window);
|
|
gdk_window_set_background (widget->window, &widget->style->base[GTK_STATE_NORMAL]);
|
|
|
|
moo_icon_view_invalidate_layout (view);
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_unrealize (GtkWidget *widget)
|
|
{
|
|
MooIconView *view = MOO_ICON_VIEW (widget);
|
|
|
|
gdk_window_set_user_data (widget->window, NULL);
|
|
gdk_window_destroy (widget->window);
|
|
widget->window = NULL;
|
|
GTK_WIDGET_UNSET_REALIZED (widget);
|
|
|
|
if (view->priv->sel_gc)
|
|
{
|
|
g_object_unref (view->priv->sel_gc);
|
|
view->priv->sel_gc = NULL;
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_size_request (G_GNUC_UNUSED GtkWidget *widget,
|
|
GtkRequisition *requisition)
|
|
{
|
|
requisition->width = 1;
|
|
requisition->height = 1;
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_size_allocate (GtkWidget *widget,
|
|
GtkAllocation *allocation)
|
|
{
|
|
gboolean height_changed = FALSE;
|
|
MooIconView *view = MOO_ICON_VIEW (widget);
|
|
|
|
if (GTK_WIDGET_REALIZED (widget))
|
|
{
|
|
if (widget->allocation.height < 0 ||
|
|
view->priv->layout->row_height == 0)
|
|
{
|
|
height_changed = TRUE;
|
|
}
|
|
else
|
|
{
|
|
height_changed =
|
|
(allocation->height/view->priv->layout->row_height !=
|
|
view->priv->layout->num_rows);
|
|
}
|
|
}
|
|
|
|
widget->allocation = *allocation;
|
|
|
|
if (GTK_WIDGET_REALIZED (widget))
|
|
{
|
|
gdk_window_move_resize (widget->window,
|
|
allocation->x,
|
|
allocation->y,
|
|
allocation->width,
|
|
allocation->height);
|
|
}
|
|
|
|
if (height_changed)
|
|
moo_icon_view_invalidate_layout (view);
|
|
else
|
|
moo_icon_view_update_adjustment (view);
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_invalidate_layout (MooIconView *view)
|
|
{
|
|
if (!view->priv->update_idle)
|
|
view->priv->update_idle =
|
|
gdk_threads_add_idle_full (G_PRIORITY_HIGH,
|
|
(GSourceFunc) moo_icon_view_update_layout,
|
|
view, NULL);
|
|
}
|
|
|
|
|
|
static void
|
|
init_layout (MooIconView *view)
|
|
{
|
|
view->priv->layout = g_new0 (Layout, 1);
|
|
}
|
|
|
|
|
|
static void
|
|
destroy_layout (MooIconView *view)
|
|
{
|
|
GSList *l;
|
|
|
|
for (l = view->priv->layout->columns; l != NULL; l = l->next)
|
|
{
|
|
Column *column = reinterpret_cast<Column*> (l->data);
|
|
gtk_tree_path_free (column->first);
|
|
g_ptr_array_free (column->entries, TRUE);
|
|
g_free (column);
|
|
}
|
|
|
|
g_slist_free (view->priv->layout->columns);
|
|
g_free (view->priv->layout);
|
|
view->priv->layout = NULL;
|
|
}
|
|
|
|
|
|
static int
|
|
num_entries (Column *column)
|
|
{
|
|
g_assert (column != NULL);
|
|
return column->entries->len;
|
|
}
|
|
|
|
|
|
static void draw_column (MooIconView *view,
|
|
Column *column,
|
|
GdkRegion *clip);
|
|
static void draw_entry (MooIconView *view,
|
|
GtkTreeIter *iter,
|
|
GtkTreePath *path,
|
|
GdkRectangle *entry_rect);
|
|
|
|
static void
|
|
get_rect_from_points (GdkRectangle *rect,
|
|
int x1,
|
|
int y1,
|
|
int x2,
|
|
int y2)
|
|
{
|
|
rect->x = MIN (x1, x2);
|
|
rect->y = MIN (y1, y2);
|
|
rect->width = MAX (x1, x2) - rect->x + 1;
|
|
rect->height = MAX (y1, y2) - rect->y + 1;
|
|
}
|
|
|
|
static void
|
|
get_drag_select_rect (MooIconView *view,
|
|
GdkRectangle *rect)
|
|
{
|
|
get_rect_from_points (rect,
|
|
view->priv->drag_select_x - view->priv->xoffset,
|
|
view->priv->drag_select_y,
|
|
view->priv->button_press_x - view->priv->xoffset,
|
|
view->priv->button_press_y);
|
|
}
|
|
|
|
static gboolean
|
|
moo_icon_view_expose (GtkWidget *widget,
|
|
GdkEventExpose *event)
|
|
{
|
|
GSList *l;
|
|
GdkRegion *area;
|
|
MooIconView *view = MOO_ICON_VIEW (widget);
|
|
Layout *layout = view->priv->layout;
|
|
|
|
if (check_empty (view))
|
|
return TRUE;
|
|
|
|
area = gdk_region_copy (event->region);
|
|
gdk_region_offset (area, view->priv->xoffset, 0);
|
|
|
|
for (l = layout->columns; l != NULL; l = l->next)
|
|
{
|
|
GdkRectangle column_rect;
|
|
GdkRegion *column_region;
|
|
Column *column;
|
|
|
|
column = reinterpret_cast<Column*> (l->data);
|
|
|
|
column_rect.x = column->offset;
|
|
column_rect.y = 0;
|
|
column_rect.width = column->width;
|
|
column_rect.height = num_entries (column) * layout->row_height;
|
|
|
|
column_region = gdk_region_rectangle (&column_rect);
|
|
gdk_region_intersect (column_region, area);
|
|
|
|
if (!gdk_region_empty (column_region))
|
|
{
|
|
draw_column (view, column, column_region);
|
|
}
|
|
|
|
gdk_region_destroy (column_region);
|
|
}
|
|
|
|
gdk_region_destroy (area);
|
|
|
|
if (view->priv->drag_select)
|
|
{
|
|
cairo_t *cr;
|
|
GdkRectangle rect;
|
|
GdkColor *color;
|
|
double dash_len = 1.;
|
|
|
|
cr = gdk_cairo_create (event->window);
|
|
get_drag_select_rect (view, &rect);
|
|
|
|
color = &widget->style->base[GTK_STATE_SELECTED];
|
|
|
|
cairo_set_source_rgba (cr,
|
|
color->red / 65535.,
|
|
color->green / 65535.,
|
|
color->blue / 65535.,
|
|
1 / 3.);
|
|
gdk_cairo_rectangle (cr, &rect);
|
|
cairo_fill (cr);
|
|
|
|
cairo_set_dash (cr, &dash_len, 1, .5);
|
|
cairo_set_line_width (cr, 1.);
|
|
gdk_cairo_set_source_color (cr, color);
|
|
cairo_rectangle (cr,
|
|
rect.x + .5,
|
|
rect.y + .5,
|
|
rect.width - 1,
|
|
rect.height - 1);
|
|
cairo_stroke (cr);
|
|
|
|
cairo_destroy (cr);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static void draw_column (MooIconView *view,
|
|
Column *column,
|
|
GdkRegion *clip)
|
|
{
|
|
int i;
|
|
GdkRectangle clip_rect;
|
|
GtkTreeIter iter;
|
|
GtkTreePath *path;
|
|
Layout *layout = view->priv->layout;
|
|
|
|
gdk_region_get_clipbox (clip, &clip_rect);
|
|
|
|
gtk_tree_model_get_iter (view->priv->model, &iter,
|
|
column->first);
|
|
path = gtk_tree_path_copy (column->first);
|
|
|
|
for (i = 0; i < num_entries (column); ++i)
|
|
{
|
|
GdkRectangle entry_rect, dummy;
|
|
|
|
entry_rect.x = column->offset;
|
|
entry_rect.y = i * layout->row_height;
|
|
entry_rect.width = layout->pixbuf_width +
|
|
GPOINTER_TO_INT (column->entries->pdata[i]);
|
|
entry_rect.height = layout->row_height;
|
|
|
|
if (gdk_rectangle_intersect (&entry_rect,
|
|
&clip_rect,
|
|
&dummy))
|
|
{
|
|
entry_rect.x -= view->priv->xoffset;
|
|
draw_entry (view, &iter, path, &entry_rect);
|
|
}
|
|
|
|
gtk_tree_model_iter_next (view->priv->model, &iter);
|
|
gtk_tree_path_next (path);
|
|
}
|
|
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
|
|
static void draw_entry (MooIconView *view,
|
|
GtkTreeIter *iter,
|
|
GtkTreePath *path,
|
|
GdkRectangle *entry_rect)
|
|
{
|
|
GtkWidget *widget = GTK_WIDGET (view);
|
|
GdkRectangle cell_area = *entry_rect;
|
|
GtkCellRendererState state = GtkCellRendererState (0);
|
|
GtkTreePath *cursor_path, *drop_path;
|
|
gboolean selected, cursor, drop;
|
|
|
|
selected = _moo_icon_view_path_is_selected (view, path);
|
|
|
|
cursor_path = _moo_icon_view_get_cursor (view);
|
|
cursor = cursor_path != NULL && !gtk_tree_path_compare (cursor_path, path);
|
|
gtk_tree_path_free (cursor_path);
|
|
|
|
drop_path = _moo_icon_view_get_drag_dest_row (view);
|
|
drop = drop_path != NULL && !gtk_tree_path_compare (drop_path, path);
|
|
gtk_tree_path_free (drop_path);
|
|
|
|
if (selected || drop)
|
|
{
|
|
GdkGC *selection_gc;
|
|
|
|
if (GTK_WIDGET_HAS_FOCUS (widget) || drop)
|
|
{
|
|
selection_gc = widget->style->base_gc [GTK_STATE_SELECTED];
|
|
state = GTK_CELL_RENDERER_SELECTED | GTK_CELL_RENDERER_FOCUSED;
|
|
}
|
|
else
|
|
{
|
|
selection_gc = widget->style->base_gc [GTK_STATE_ACTIVE];
|
|
state = GTK_CELL_RENDERER_SELECTED;
|
|
}
|
|
|
|
gdk_draw_rectangle (widget->window,
|
|
selection_gc,
|
|
TRUE,
|
|
entry_rect->x,
|
|
entry_rect->y,
|
|
entry_rect->width,
|
|
entry_rect->height);
|
|
}
|
|
|
|
if (view->priv->pixbuf.show && view->priv->layout->pixbuf_height > 0)
|
|
{
|
|
view->priv->pixbuf.func (view, view->priv->pixbuf.cell,
|
|
view->priv->model, iter,
|
|
view->priv->pixbuf.func_data);
|
|
|
|
cell_area.width = view->priv->layout->pixbuf_width;
|
|
|
|
gtk_cell_renderer_render (view->priv->pixbuf.cell,
|
|
widget->window, widget,
|
|
entry_rect,
|
|
&cell_area,
|
|
entry_rect,
|
|
state);
|
|
}
|
|
|
|
if (view->priv->text.show && view->priv->layout->text_height > 0)
|
|
{
|
|
view->priv->text.func (view, view->priv->text.cell,
|
|
view->priv->model, iter,
|
|
view->priv->text.func_data);
|
|
|
|
cell_area.x += view->priv->layout->pixbuf_width;
|
|
cell_area.width = entry_rect->width - view->priv->layout->pixbuf_width;
|
|
|
|
gtk_cell_renderer_render (view->priv->text.cell,
|
|
widget->window, widget,
|
|
entry_rect,
|
|
&cell_area,
|
|
entry_rect,
|
|
state);
|
|
}
|
|
|
|
if (cursor || drop)
|
|
{
|
|
gtk_paint_focus (widget->style,
|
|
widget->window,
|
|
GTK_STATE_SELECTED,
|
|
entry_rect,
|
|
widget,
|
|
"icon_view",
|
|
entry_rect->x,
|
|
entry_rect->y,
|
|
entry_rect->width,
|
|
entry_rect->height);
|
|
}
|
|
}
|
|
|
|
|
|
static void cell_data_func (G_GNUC_UNUSED MooIconView *view,
|
|
GtkCellRenderer *cell,
|
|
GtkTreeModel *model,
|
|
GtkTreeIter *iter,
|
|
gpointer cell_info)
|
|
{
|
|
GSList *l;
|
|
CellInfo *info = reinterpret_cast<CellInfo*> (cell_info);
|
|
static GValue value;
|
|
|
|
for (l = info->attributes; l && l->next; l = l->next->next)
|
|
{
|
|
gtk_tree_model_get_value (model, iter,
|
|
GPOINTER_TO_INT (l->next->data),
|
|
&value);
|
|
g_object_set_property (G_OBJECT (cell), (char*)l->data, &value);
|
|
g_value_unset (&value);
|
|
}
|
|
}
|
|
|
|
|
|
/****************************************************************************/
|
|
/* Creating layout
|
|
*/
|
|
|
|
static gboolean model_empty (GtkTreeModel *model)
|
|
{
|
|
GtkTreeIter iter;
|
|
return !gtk_tree_model_get_iter_first (model, &iter);
|
|
}
|
|
|
|
|
|
static void calculate_pixbuf_size (MooIconView *view);
|
|
static void calculate_row_height (MooIconView *view);
|
|
static gboolean calculate_column_width (MooIconView *view,
|
|
GtkTreeIter *iter,
|
|
Column *column);
|
|
|
|
|
|
static gboolean moo_icon_view_update_layout (MooIconView *view)
|
|
{
|
|
GtkWidget *widget = GTK_WIDGET (view);
|
|
Layout *layout = view->priv->layout;
|
|
GtkTreeModel *model = view->priv->model;
|
|
GtkTreeIter iter;
|
|
gboolean finish;
|
|
GSList *l;
|
|
int column_index;
|
|
|
|
view->priv->update_idle = 0;
|
|
gtk_widget_queue_draw (GTK_WIDGET (view));
|
|
|
|
for (l = layout->columns; l != NULL; l = l->next)
|
|
{
|
|
Column *column = reinterpret_cast<Column*> (l->data);
|
|
gtk_tree_path_free (column->first);
|
|
g_ptr_array_free (column->entries, TRUE);
|
|
g_free (column);
|
|
}
|
|
|
|
g_slist_free (layout->columns);
|
|
layout->columns = NULL;
|
|
layout->num_rows = 0;
|
|
|
|
layout->width = 0;
|
|
layout->height = 0;
|
|
layout->row_height = 0;
|
|
|
|
layout->pixbuf_width = 0;
|
|
layout->pixbuf_height = 0;
|
|
layout->text_height = 0;
|
|
|
|
if (!GTK_WIDGET_REALIZED (view) ||
|
|
!GTK_WIDGET_MAPPED (view) ||
|
|
!view->priv->model ||
|
|
model_empty (view->priv->model))
|
|
{
|
|
view->priv->xoffset = 0;
|
|
moo_icon_view_update_adjustment (view);
|
|
return FALSE;
|
|
}
|
|
|
|
calculate_pixbuf_size (view);
|
|
calculate_row_height (view);
|
|
|
|
layout->num_rows = MAX (widget->allocation.height / layout->row_height, 1);
|
|
layout->height = layout->row_height * layout->num_rows;
|
|
layout->width = 0;
|
|
|
|
gtk_tree_model_get_iter_first (model, &iter);
|
|
finish = FALSE;
|
|
column_index = 0;
|
|
|
|
while (!finish)
|
|
{
|
|
Column *column = g_new0 (Column, 1);
|
|
|
|
column->index = column_index++;
|
|
column->offset = layout->width;
|
|
column->entries = g_ptr_array_new ();
|
|
|
|
column->first = gtk_tree_model_get_path (model, &iter);
|
|
|
|
finish = !calculate_column_width (view, &iter, column);
|
|
|
|
layout->width += column->width;
|
|
layout->columns = g_slist_append (layout->columns, column);
|
|
}
|
|
|
|
if (view->priv->scroll_to)
|
|
{
|
|
if (gtk_tree_row_reference_valid (view->priv->scroll_to))
|
|
{
|
|
GtkTreePath *path = gtk_tree_row_reference_get_path (view->priv->scroll_to);
|
|
moo_icon_view_update_adjustment (view);
|
|
_moo_icon_view_scroll_to_cell (view, path);
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
gtk_tree_row_reference_free (view->priv->scroll_to);
|
|
view->priv->scroll_to = NULL;
|
|
}
|
|
else
|
|
{
|
|
view->priv->xoffset = clamp_offset (view, view->priv->xoffset);
|
|
moo_icon_view_update_adjustment (view);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static void set_pixbuf_size (MooIconView *view,
|
|
int width,
|
|
int height)
|
|
{
|
|
if (width < 0) width = 0;
|
|
if (height < 0) height = 0;
|
|
|
|
if (!width || !height)
|
|
{
|
|
view->priv->layout->pixbuf_width = 0;
|
|
view->priv->layout->pixbuf_height = 0;
|
|
}
|
|
else
|
|
{
|
|
view->priv->layout->pixbuf_width = width;
|
|
view->priv->layout->pixbuf_height = height;
|
|
}
|
|
}
|
|
|
|
|
|
static void calculate_pixbuf_size (MooIconView *view)
|
|
{
|
|
GtkTreeIter iter;
|
|
int width, height;
|
|
|
|
if (!view->priv->pixbuf.show)
|
|
{
|
|
set_pixbuf_size (view, 0, 0);
|
|
return;
|
|
}
|
|
|
|
if (view->priv->pixel_icon_size >= 0)
|
|
{
|
|
set_pixbuf_size (view, view->priv->pixel_icon_size,
|
|
view->priv->pixel_icon_size);
|
|
return;
|
|
}
|
|
|
|
if (view->priv->icon_size >= 0 &&
|
|
gtk_icon_size_lookup (GtkIconSize (view->priv->icon_size), &width, &height))
|
|
{
|
|
set_pixbuf_size (view, width, height);
|
|
return;
|
|
}
|
|
|
|
g_object_get (view->priv->pixbuf.cell,
|
|
"height", &height,
|
|
"width", &width,
|
|
NULL);
|
|
|
|
if (height >= 0 && width >= 0)
|
|
{
|
|
set_pixbuf_size (view, width, height);
|
|
return;
|
|
}
|
|
|
|
gtk_tree_model_get_iter_first (view->priv->model, &iter);
|
|
view->priv->pixbuf.func (view, view->priv->pixbuf.cell,
|
|
view->priv->model, &iter,
|
|
view->priv->pixbuf.func_data);
|
|
gtk_cell_renderer_get_size (view->priv->pixbuf.cell,
|
|
GTK_WIDGET (view), NULL,
|
|
NULL, NULL, &width, &height);
|
|
|
|
set_pixbuf_size (view, width, height);
|
|
}
|
|
|
|
|
|
static void set_text_height (MooIconView *view,
|
|
int height)
|
|
{
|
|
if (height < 0) height = 0;
|
|
view->priv->layout->text_height = height;
|
|
view->priv->layout->row_height =
|
|
MAX (view->priv->layout->pixbuf_height, height);
|
|
}
|
|
|
|
|
|
static void calculate_row_height (MooIconView *view)
|
|
{
|
|
GtkTreeIter iter;
|
|
int height;
|
|
|
|
if (!view->priv->text.show)
|
|
{
|
|
set_text_height (view, 0);
|
|
return;
|
|
}
|
|
|
|
g_object_get (view->priv->text.cell,
|
|
"height", &height,
|
|
NULL);
|
|
|
|
if (height >= 0)
|
|
{
|
|
set_text_height (view, height);
|
|
return;
|
|
}
|
|
|
|
gtk_tree_model_get_iter_first (view->priv->model, &iter);
|
|
view->priv->text.func (view, view->priv->text.cell,
|
|
view->priv->model, &iter,
|
|
view->priv->text.func_data);
|
|
gtk_cell_renderer_get_size (view->priv->text.cell,
|
|
GTK_WIDGET (view), NULL,
|
|
NULL, NULL, NULL, &height);
|
|
|
|
set_text_height (view, height);
|
|
}
|
|
|
|
|
|
static gboolean calculate_column_width (MooIconView *view,
|
|
GtkTreeIter *iter,
|
|
Column *column)
|
|
{
|
|
Layout *layout = view->priv->layout;
|
|
int max_num = layout->num_rows;
|
|
GtkWidget *widget = GTK_WIDGET (view);
|
|
gboolean use_text = (layout->text_height > 0);
|
|
GtkTreeModel *model = view->priv->model;
|
|
|
|
g_ptr_array_set_size (column->entries, 0);
|
|
column->width = 0;
|
|
|
|
while (TRUE)
|
|
{
|
|
g_ptr_array_add (column->entries, NULL);
|
|
|
|
if (use_text)
|
|
{
|
|
int text_width;
|
|
|
|
view->priv->text.func (view, view->priv->text.cell,
|
|
model, iter,
|
|
view->priv->text.func_data);
|
|
gtk_cell_renderer_get_size (view->priv->text.cell,
|
|
widget, NULL, NULL, NULL,
|
|
&text_width, NULL);
|
|
if (column->width < text_width)
|
|
column->width = text_width;
|
|
|
|
column->entries->pdata[num_entries (column) - 1] =
|
|
GINT_TO_POINTER (text_width);
|
|
}
|
|
|
|
if (num_entries (column) == max_num)
|
|
{
|
|
column->width += layout->pixbuf_width;
|
|
return gtk_tree_model_iter_next (model, iter);
|
|
}
|
|
else
|
|
{
|
|
if (!gtk_tree_model_iter_next (model, iter))
|
|
{
|
|
column->width += layout->pixbuf_width;
|
|
return FALSE;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static int path_get_index (GtkTreePath *path)
|
|
{
|
|
g_assert (gtk_tree_path_get_depth (path) == 1);
|
|
return gtk_tree_path_get_indices(path)[0];
|
|
}
|
|
|
|
|
|
static Column *find_column_by_path (MooIconView *view,
|
|
GtkTreePath *path,
|
|
int *index_)
|
|
{
|
|
GSList *l;
|
|
int path_index;
|
|
|
|
path_index = path_get_index (path);
|
|
|
|
for (l = view->priv->layout->columns; l != NULL; l = l->next)
|
|
{
|
|
Column *column = reinterpret_cast<Column*> (l->data);
|
|
int first = path_get_index (column->first);
|
|
if (first <= path_index && path_index < first +
|
|
num_entries (column))
|
|
{
|
|
if (index_)
|
|
*index_ = path_index - first;
|
|
return column;
|
|
}
|
|
}
|
|
|
|
g_return_val_if_reached (NULL);
|
|
}
|
|
|
|
|
|
static void row_changed (G_GNUC_UNUSED GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
G_GNUC_UNUSED GtkTreeIter *iter,
|
|
MooIconView *view)
|
|
{
|
|
if (!GTK_WIDGET_REALIZED (view) ||
|
|
!GTK_WIDGET_MAPPED (view))
|
|
return;
|
|
|
|
if (gtk_tree_path_get_depth (path) != 1)
|
|
return;
|
|
|
|
drag_scroll_stop (view);
|
|
moo_icon_view_invalidate_layout (view);
|
|
}
|
|
|
|
|
|
static void row_deleted (G_GNUC_UNUSED GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
MooIconView *view)
|
|
{
|
|
if (gtk_tree_path_get_depth (path) != 1)
|
|
return;
|
|
|
|
selection_row_deleted (view);
|
|
cursor_row_deleted (view);
|
|
|
|
if (!GTK_WIDGET_REALIZED (view) ||
|
|
!GTK_WIDGET_MAPPED (view))
|
|
return;
|
|
|
|
drag_scroll_stop (view);
|
|
moo_icon_view_invalidate_layout (view);
|
|
}
|
|
|
|
|
|
static void row_inserted (G_GNUC_UNUSED GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
G_GNUC_UNUSED GtkTreeIter *iter,
|
|
MooIconView *view)
|
|
{
|
|
if (!GTK_WIDGET_REALIZED (view) ||
|
|
!GTK_WIDGET_MAPPED (view))
|
|
return;
|
|
|
|
if (gtk_tree_path_get_depth (path) != 1)
|
|
return;
|
|
|
|
drag_scroll_stop (view);
|
|
moo_icon_view_invalidate_layout (view);
|
|
}
|
|
|
|
|
|
static void rows_reordered (G_GNUC_UNUSED GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
G_GNUC_UNUSED GtkTreeIter *iter,
|
|
G_GNUC_UNUSED gpointer whatever,
|
|
MooIconView *view)
|
|
{
|
|
if (!GTK_WIDGET_REALIZED (view) ||
|
|
!GTK_WIDGET_MAPPED (view))
|
|
return;
|
|
|
|
if (gtk_tree_path_get_depth (path) != 0)
|
|
return;
|
|
|
|
drag_scroll_stop (view);
|
|
moo_icon_view_invalidate_layout (view);
|
|
}
|
|
|
|
|
|
static void invalidate_cell_rect (MooIconView *view,
|
|
Column *column,
|
|
int index_)
|
|
{
|
|
GdkRectangle rect;
|
|
|
|
if (!GTK_WIDGET_REALIZED (view) || view->priv->update_idle)
|
|
return;
|
|
|
|
rect.x = column->offset - view->priv->xoffset;
|
|
rect.y = index_ * view->priv->layout->row_height;
|
|
rect.width = column->width;
|
|
rect.height = view->priv->layout->row_height;
|
|
|
|
gdk_window_invalidate_rect (GTK_WIDGET(view)->window, &rect, FALSE);
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_set_scroll_adjustments (GtkWidget *widget,
|
|
GtkAdjustment *hadj,
|
|
G_GNUC_UNUSED GtkAdjustment *vadj)
|
|
{
|
|
_moo_icon_view_set_adjustment (MOO_ICON_VIEW (widget), hadj);
|
|
}
|
|
|
|
|
|
static void value_changed (MooIconView *view,
|
|
GtkAdjustment *adj)
|
|
{
|
|
if (adj->value != view->priv->xoffset)
|
|
moo_icon_view_scroll_to (view, (int) adj->value);
|
|
}
|
|
|
|
|
|
static void moo_icon_view_update_adjustment (MooIconView *view)
|
|
{
|
|
GSList *link;
|
|
|
|
link = g_slist_last (view->priv->layout->columns);
|
|
view->priv->xoffset = clamp_offset (view, view->priv->xoffset);
|
|
|
|
if (!link || view->priv->layout->width <= GTK_WIDGET(view)->allocation.width)
|
|
{
|
|
view->priv->adjustment->lower = 0;
|
|
view->priv->adjustment->upper = GTK_WIDGET(view)->allocation.width - 1;
|
|
view->priv->adjustment->value = 0;
|
|
view->priv->adjustment->step_increment = GTK_WIDGET(view)->allocation.width - 1;
|
|
view->priv->adjustment->page_increment = GTK_WIDGET(view)->allocation.width - 1;
|
|
view->priv->adjustment->page_size = GTK_WIDGET(view)->allocation.width - 1;
|
|
}
|
|
else
|
|
{
|
|
Column *column = reinterpret_cast<Column*> (link->data);
|
|
|
|
view->priv->adjustment->lower = 0;
|
|
view->priv->adjustment->upper =
|
|
view->priv->layout->width - 1;
|
|
|
|
view->priv->adjustment->value = view->priv->xoffset;
|
|
|
|
view->priv->adjustment->page_increment = GTK_WIDGET(view)->allocation.width;
|
|
view->priv->adjustment->step_increment =
|
|
view->priv->layout->width / (column->index + 1);
|
|
|
|
view->priv->adjustment->page_size = GTK_WIDGET(view)->allocation.width;
|
|
}
|
|
|
|
gtk_adjustment_changed (view->priv->adjustment);
|
|
}
|
|
|
|
|
|
static void
|
|
_moo_icon_view_set_adjustment (MooIconView *view,
|
|
GtkAdjustment *adjustment)
|
|
{
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (!adjustment || GTK_IS_ADJUSTMENT (adjustment));
|
|
|
|
if (!adjustment)
|
|
adjustment = GTK_ADJUSTMENT (gtk_adjustment_new (0,0,0,0,0,0));
|
|
|
|
if (view->priv->adjustment)
|
|
{
|
|
g_signal_handlers_disconnect_by_func (view->priv->adjustment,
|
|
(gpointer) value_changed,
|
|
view);
|
|
g_object_unref (view->priv->adjustment);
|
|
}
|
|
|
|
view->priv->adjustment = adjustment;
|
|
g_object_ref_sink (adjustment);
|
|
|
|
g_signal_connect_swapped (adjustment, "value-changed",
|
|
G_CALLBACK (value_changed),
|
|
view);
|
|
|
|
if (GTK_WIDGET_REALIZED (view) && GTK_WIDGET_MAPPED (view))
|
|
moo_icon_view_update_adjustment (view);
|
|
}
|
|
|
|
|
|
static void
|
|
cleanup_after_button_press (MooIconView *view)
|
|
{
|
|
if (view->priv->button_press_row)
|
|
{
|
|
gtk_tree_row_reference_free (view->priv->button_press_row);
|
|
view->priv->button_press_row = NULL;
|
|
}
|
|
|
|
view->priv->button_pressed = 0;
|
|
drag_select_finish (view);
|
|
drag_scroll_stop (view);
|
|
}
|
|
|
|
static gboolean
|
|
moo_icon_view_button_press (GtkWidget *widget,
|
|
GdkEventButton *event)
|
|
{
|
|
MooIconView *view = MOO_ICON_VIEW (widget);
|
|
GtkTreePath *path = NULL;
|
|
GdkModifierType mods = GdkModifierType (event->state & gtk_accelerator_get_default_mod_mask ());
|
|
|
|
view->priv->button_pressed = 0;
|
|
|
|
if (event->button == 1)
|
|
{
|
|
gtk_widget_grab_focus (widget);
|
|
_moo_icon_view_get_path_at_pos (view, (int) event->x, (int) event->y,
|
|
&path, NULL, NULL, NULL);
|
|
|
|
switch (event->type)
|
|
{
|
|
case GDK_BUTTON_PRESS:
|
|
if (path && _moo_icon_view_path_is_selected (view, path))
|
|
{
|
|
view->priv->button_press_row =
|
|
gtk_tree_row_reference_new (view->priv->model, path);
|
|
gtk_tree_path_free (path);
|
|
}
|
|
else if (path)
|
|
{
|
|
if (mods & GDK_SHIFT_MASK)
|
|
{
|
|
GtkTreePath *cursor_path = ensure_cursor (view);
|
|
_moo_icon_view_unselect_all (view);
|
|
moo_icon_view_select_range (view, path, cursor_path);
|
|
}
|
|
else if (mods & GDK_CONTROL_MASK)
|
|
{
|
|
moo_icon_view_select_path (view, path);
|
|
}
|
|
else
|
|
{
|
|
_moo_icon_view_unselect_all (view);
|
|
_moo_icon_view_set_cursor (view, path, FALSE);
|
|
}
|
|
|
|
gtk_tree_path_free (path);
|
|
}
|
|
else if (!(mods & (GDK_SHIFT_MASK | GDK_CONTROL_MASK)))
|
|
{
|
|
_moo_icon_view_unselect_all (view);
|
|
}
|
|
|
|
/* this is later checked in maybe_drag */
|
|
view->priv->button_pressed = event->button;
|
|
view->priv->button_press_mods = mods;
|
|
view->priv->button_press_x = (int) event->x + view->priv->xoffset;
|
|
view->priv->button_press_y = (int) event->y;
|
|
|
|
return TRUE;
|
|
|
|
case GDK_2BUTTON_PRESS:
|
|
cleanup_after_button_press (view);
|
|
if (path)
|
|
{
|
|
activate_item_at_cursor (view);
|
|
gtk_tree_path_free (path);
|
|
}
|
|
return TRUE;
|
|
|
|
default:
|
|
cleanup_after_button_press (view);
|
|
gtk_tree_path_free (path);
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
moo_icon_view_motion_notify (GtkWidget *widget,
|
|
GdkEventMotion *event)
|
|
{
|
|
MooIconView *view = MOO_ICON_VIEW (widget);
|
|
|
|
if (moo_icon_view_maybe_drag_select (view, event))
|
|
return TRUE;
|
|
|
|
drag_scroll_stop (view);
|
|
return moo_icon_view_maybe_drag (view, event);
|
|
}
|
|
|
|
|
|
static void
|
|
drag_select_finish (MooIconView *view)
|
|
{
|
|
if (view->priv->drag_select)
|
|
{
|
|
if (GTK_WIDGET_DRAWABLE (view))
|
|
{
|
|
GdkRectangle rect;
|
|
get_drag_select_rect (view, &rect);
|
|
gdk_window_invalidate_rect (GTK_WIDGET (view)->window, &rect, TRUE);
|
|
}
|
|
|
|
if (view->priv->old_selection)
|
|
g_tree_destroy (view->priv->old_selection);
|
|
view->priv->old_selection = NULL;
|
|
|
|
view->priv->drag_select = FALSE;
|
|
}
|
|
}
|
|
|
|
static gboolean
|
|
moo_icon_view_button_release (GtkWidget *widget,
|
|
G_GNUC_UNUSED GdkEventButton *event)
|
|
{
|
|
MooIconView *view = MOO_ICON_VIEW (widget);
|
|
GtkTreePath *path = NULL;
|
|
|
|
if (view->priv->button_press_row)
|
|
path = gtk_tree_row_reference_get_path (view->priv->button_press_row);
|
|
|
|
if (path)
|
|
{
|
|
if (view->priv->button_press_mods & GDK_SHIFT_MASK)
|
|
{
|
|
GtkTreePath *cursor_path = ensure_cursor (view);
|
|
_moo_icon_view_unselect_all (view);
|
|
moo_icon_view_select_range (view, path, cursor_path);
|
|
}
|
|
else if (view->priv->button_press_mods & GDK_CONTROL_MASK)
|
|
{
|
|
if (_moo_icon_view_path_is_selected (view, path))
|
|
moo_icon_view_unselect_path (view, path);
|
|
else
|
|
moo_icon_view_select_path (view, path);
|
|
}
|
|
else
|
|
{
|
|
_moo_icon_view_unselect_all (view);
|
|
_moo_icon_view_set_cursor (view, path, FALSE);
|
|
}
|
|
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
cleanup_after_button_press (view);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
traverse_tree_prepend_path (gpointer key,
|
|
G_GNUC_UNUSED gpointer dummy,
|
|
GList **list)
|
|
{
|
|
*list = g_list_prepend (*list, key);
|
|
return FALSE;
|
|
}
|
|
|
|
static GTree *
|
|
path_set_new (void)
|
|
{
|
|
return g_tree_new_full ((GCompareDataFunc) gtk_tree_path_compare, NULL,
|
|
(GDestroyNotify) gtk_tree_path_free, NULL);
|
|
}
|
|
|
|
static GTree *
|
|
path_set_from_list (GList *list)
|
|
{
|
|
GTree *tree;
|
|
|
|
if (!list)
|
|
return NULL;
|
|
|
|
tree = path_set_new ();
|
|
|
|
while (list)
|
|
{
|
|
GtkTreePath *path = reinterpret_cast<GtkTreePath*> (list->data);
|
|
g_tree_replace (tree, path, path);
|
|
list = g_list_delete_link (list, list);
|
|
}
|
|
|
|
return tree;
|
|
}
|
|
|
|
static gboolean
|
|
traverse_tree_copy_path (GtkTreePath *path,
|
|
G_GNUC_UNUSED gpointer dummy,
|
|
GTree *tree)
|
|
{
|
|
GtkTreePath *copy = gtk_tree_path_copy (path);
|
|
g_tree_replace (tree, copy, copy);
|
|
return FALSE;
|
|
}
|
|
|
|
static GTree *
|
|
path_set_union (GTree *tree,
|
|
GTree *second)
|
|
{
|
|
if (!second || !g_tree_height (second))
|
|
return tree;
|
|
|
|
if (!tree)
|
|
tree = path_set_new ();
|
|
|
|
g_tree_foreach (second, (GTraverseFunc) traverse_tree_copy_path, tree);
|
|
|
|
return tree;
|
|
}
|
|
|
|
static GTree *
|
|
path_set_copy (GTree *tree)
|
|
{
|
|
return path_set_union (NULL, tree);
|
|
}
|
|
|
|
static gboolean
|
|
traverse_tree_remove_path (GtkTreePath *path,
|
|
G_GNUC_UNUSED gpointer dummy,
|
|
GTree *tree)
|
|
{
|
|
g_tree_remove (tree, path);
|
|
return FALSE;
|
|
}
|
|
|
|
static GTree *
|
|
path_set_difference (GTree *tree,
|
|
GTree *second)
|
|
{
|
|
if (!tree || !g_tree_height (tree) || !second || !g_tree_height (second))
|
|
return tree;
|
|
|
|
g_tree_foreach (second, (GTraverseFunc) traverse_tree_remove_path, tree);
|
|
|
|
if (!g_tree_height (tree))
|
|
{
|
|
g_tree_destroy (tree);
|
|
tree = NULL;
|
|
}
|
|
|
|
return tree;
|
|
}
|
|
|
|
static void
|
|
path_set_free (GTree *tree)
|
|
{
|
|
if (tree)
|
|
g_tree_destroy (tree);
|
|
}
|
|
|
|
static void
|
|
moo_icon_view_drag_select (MooIconView *view,
|
|
GdkEventMotion *event)
|
|
{
|
|
GdkRectangle rect;
|
|
GdkRegion *region;
|
|
GList *rect_items_list;
|
|
GTree *new_selection;
|
|
|
|
region = gdk_region_new ();
|
|
|
|
get_drag_select_rect (view, &rect);
|
|
gdk_region_union_with_rect (region, &rect);
|
|
|
|
view->priv->drag_select_x = event->x + view->priv->xoffset;
|
|
view->priv->drag_select_y = event->y;
|
|
get_drag_select_rect (view, &rect);
|
|
gdk_region_union_with_rect (region, &rect);
|
|
|
|
gdk_window_invalidate_region (GTK_WIDGET (view)->window, region, TRUE);
|
|
drag_scroll_check (view, (int) event->x, (int) event->y);
|
|
|
|
rect_items_list = moo_icon_view_get_paths_in_rect (view, &rect);
|
|
new_selection = path_set_from_list (rect_items_list);
|
|
|
|
if (view->priv->button_press_mods & GDK_SHIFT_MASK)
|
|
{
|
|
new_selection = path_set_union (new_selection, view->priv->old_selection);
|
|
}
|
|
else if (view->priv->button_press_mods & GDK_CONTROL_MASK)
|
|
{
|
|
GTree *old_to_select;
|
|
old_to_select = path_set_copy (view->priv->old_selection);
|
|
old_to_select = path_set_difference (old_to_select, new_selection);
|
|
new_selection = path_set_difference (new_selection, view->priv->old_selection);
|
|
new_selection = path_set_union (new_selection, old_to_select);
|
|
path_set_free (old_to_select);
|
|
}
|
|
|
|
_moo_icon_view_unselect_all (view);
|
|
|
|
if (new_selection)
|
|
{
|
|
GList *list = NULL;
|
|
g_tree_foreach (new_selection,
|
|
(GTraverseFunc) traverse_tree_prepend_path,
|
|
&list);
|
|
moo_icon_view_select_paths (view, list);
|
|
g_list_free (list);
|
|
}
|
|
|
|
path_set_free (new_selection);
|
|
gdk_region_destroy (region);
|
|
}
|
|
|
|
static gboolean
|
|
moo_icon_view_maybe_drag_select (MooIconView *view,
|
|
GdkEventMotion *event)
|
|
{
|
|
if (!view->priv->drag_select)
|
|
{
|
|
GList *selected;
|
|
|
|
if (view->priv->button_pressed != 1)
|
|
return FALSE;
|
|
|
|
if (_moo_icon_view_get_path_at_pos (view,
|
|
view->priv->button_press_x - view->priv->xoffset,
|
|
view->priv->button_press_y,
|
|
NULL, NULL, NULL, NULL))
|
|
return FALSE;
|
|
|
|
if (!(view->priv->button_press_mods & (GDK_SHIFT_MASK | GDK_CONTROL_MASK)))
|
|
_moo_icon_view_unselect_all (view);
|
|
|
|
selected = _moo_icon_view_get_selected_rows (view);
|
|
view->priv->old_selection = path_set_from_list (selected);
|
|
view->priv->drag_select_x = view->priv->button_press_x;
|
|
view->priv->drag_select_y = view->priv->button_press_y;
|
|
view->priv->drag_select = TRUE;
|
|
}
|
|
|
|
moo_icon_view_drag_select (view, event);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static void invalidate_path_rectangle (MooIconView *view,
|
|
GtkTreePath *path)
|
|
{
|
|
Column *column;
|
|
int index_ = 0;
|
|
|
|
if (!GTK_WIDGET_REALIZED (view) || view->priv->update_idle)
|
|
return;
|
|
|
|
if (check_empty (view))
|
|
return;
|
|
|
|
column = find_column_by_path (view, path, &index_);
|
|
g_return_if_fail (column != NULL);
|
|
invalidate_cell_rect (view, column, index_);
|
|
}
|
|
|
|
|
|
static void move_cursor_right (MooIconView *view,
|
|
gboolean extend_selection);
|
|
static void move_cursor_left (MooIconView *view,
|
|
gboolean extend_selection);
|
|
static void move_cursor_up (MooIconView *view,
|
|
gboolean extend_selection);
|
|
static void move_cursor_down (MooIconView *view,
|
|
gboolean extend_selection);
|
|
static void move_cursor_page_up (MooIconView *view,
|
|
gboolean extend_selection);
|
|
static void move_cursor_page_down (MooIconView *view,
|
|
gboolean extend_selection);
|
|
static void move_cursor_home (MooIconView *view,
|
|
gboolean extend_selection);
|
|
static void move_cursor_end (MooIconView *view,
|
|
gboolean extend_selection);
|
|
|
|
static gboolean move_cursor (MooIconView *view,
|
|
GtkMovementStep step,
|
|
gint count,
|
|
gboolean extend_selection)
|
|
{
|
|
int i;
|
|
|
|
if (check_empty (view))
|
|
return TRUE;
|
|
|
|
switch (step)
|
|
{
|
|
case GTK_MOVEMENT_LOGICAL_POSITIONS: /* left/right */
|
|
case GTK_MOVEMENT_VISUAL_POSITIONS:
|
|
if (count > 0)
|
|
for (i = 0; i < count; ++i)
|
|
move_cursor_right (view, extend_selection);
|
|
else
|
|
for (i = 0; i < -count; ++i)
|
|
move_cursor_left (view, extend_selection);
|
|
return TRUE;
|
|
|
|
case GTK_MOVEMENT_DISPLAY_LINES: /* up/down */
|
|
if (count > 0)
|
|
for (i = 0; i < count; ++i)
|
|
move_cursor_down (view, extend_selection);
|
|
else
|
|
for (i = 0; i < -count; ++i)
|
|
move_cursor_up (view, extend_selection);
|
|
return TRUE;
|
|
|
|
case GTK_MOVEMENT_PAGES: /* pgup/pgdown */
|
|
if (count > 0)
|
|
for (i = 0; i < count; ++i)
|
|
move_cursor_page_down (view, extend_selection);
|
|
else
|
|
for (i = 0; i < -count; ++i)
|
|
move_cursor_page_up (view, extend_selection);
|
|
return TRUE;
|
|
|
|
case GTK_MOVEMENT_BUFFER_ENDS: /* home/end */
|
|
if (count > 0)
|
|
move_cursor_end (view, extend_selection);
|
|
else if (count < 0)
|
|
move_cursor_home (view, extend_selection);
|
|
return TRUE;
|
|
|
|
default:
|
|
g_return_val_if_reached (FALSE);
|
|
}
|
|
}
|
|
|
|
|
|
static GtkTreePath *column_get_path (Column *column,
|
|
int index_)
|
|
{
|
|
int first;
|
|
|
|
g_return_val_if_fail (column != NULL, NULL);
|
|
g_return_val_if_fail (index_ >= 0, NULL);
|
|
g_return_val_if_fail (index_ < num_entries (column), NULL);
|
|
|
|
first = gtk_tree_path_get_indices(column->first)[0];
|
|
return gtk_tree_path_new_from_indices (first + index_, -1);
|
|
}
|
|
|
|
|
|
static int get_n_columns (MooIconView *view)
|
|
{
|
|
return g_slist_length (view->priv->layout->columns);
|
|
}
|
|
|
|
static Column *get_nth_column (MooIconView *view,
|
|
int n)
|
|
{
|
|
return reinterpret_cast<Column*> (g_slist_nth_data (view->priv->layout->columns, n));
|
|
}
|
|
|
|
static Column *column_next (MooIconView *view,
|
|
Column *column)
|
|
{
|
|
g_return_val_if_fail (column != NULL, NULL);
|
|
if (column->index == get_n_columns (view) - 1)
|
|
return NULL;
|
|
else
|
|
return get_nth_column (view, column->index + 1);
|
|
}
|
|
|
|
static Column *column_prev (MooIconView *view,
|
|
Column *column)
|
|
{
|
|
g_return_val_if_fail (column != NULL, NULL);
|
|
if (column->index == 0)
|
|
return NULL;
|
|
else
|
|
return get_nth_column (view, column->index - 1);
|
|
}
|
|
|
|
|
|
static GtkTreePath *ensure_cursor (MooIconView *view)
|
|
{
|
|
if (!view->priv->cursor)
|
|
{
|
|
GtkTreePath *path = gtk_tree_path_new_from_indices (0, -1);
|
|
view->priv->cursor = gtk_tree_row_reference_new (view->priv->model, path);
|
|
return path;
|
|
}
|
|
else
|
|
{
|
|
return gtk_tree_row_reference_get_path (view->priv->cursor);
|
|
}
|
|
}
|
|
|
|
|
|
static void move_cursor_to_entry (MooIconView *view,
|
|
Column *column,
|
|
int index_,
|
|
gboolean extend_selection)
|
|
{
|
|
GtkTreePath *path = column_get_path (column, index_);
|
|
|
|
if (extend_selection)
|
|
{
|
|
GtkTreePath *cursor_path = ensure_cursor (view);
|
|
moo_icon_view_select_range (view, cursor_path, path);
|
|
gtk_tree_path_free (cursor_path);
|
|
}
|
|
else
|
|
{
|
|
_moo_icon_view_unselect_all (view);
|
|
}
|
|
|
|
_moo_icon_view_set_cursor (view, path, FALSE);
|
|
_moo_icon_view_scroll_to_cell (view, path);
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
|
|
static void move_cursor_right (MooIconView *view,
|
|
gboolean extend_selection)
|
|
{
|
|
GtkTreePath *path;
|
|
Column *column, *next;
|
|
int y;
|
|
|
|
path = ensure_cursor (view);
|
|
|
|
column = find_column_by_path (view, path, &y);
|
|
g_return_if_fail (column != NULL);
|
|
|
|
next = column_next (view, column);
|
|
|
|
if (!next)
|
|
{
|
|
move_cursor_to_entry (view, column,
|
|
num_entries (column) - 1,
|
|
extend_selection);
|
|
}
|
|
else
|
|
{
|
|
if (y >= num_entries (next))
|
|
y = num_entries (next) - 1;
|
|
move_cursor_to_entry (view, next, y,
|
|
extend_selection);
|
|
}
|
|
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
|
|
static void move_cursor_left (MooIconView *view,
|
|
gboolean extend_selection)
|
|
{
|
|
GtkTreePath *path;
|
|
Column *column, *prev;
|
|
int index_;
|
|
|
|
path = ensure_cursor (view);
|
|
|
|
column = find_column_by_path (view, path, &index_);
|
|
g_return_if_fail (column != NULL);
|
|
|
|
prev = column_prev (view, column);
|
|
|
|
if (!prev)
|
|
move_cursor_to_entry (view, column, 0,
|
|
extend_selection);
|
|
else
|
|
move_cursor_to_entry (view, prev, index_,
|
|
extend_selection);
|
|
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
|
|
static void move_cursor_up (MooIconView *view,
|
|
gboolean extend_selection)
|
|
{
|
|
GtkTreePath *path;
|
|
Column *column, *prev;
|
|
int y;
|
|
|
|
path = ensure_cursor (view);
|
|
|
|
column = find_column_by_path (view, path, &y);
|
|
g_return_if_fail (column != NULL);
|
|
|
|
if (y)
|
|
{
|
|
move_cursor_to_entry (view, column, y - 1,
|
|
extend_selection);
|
|
}
|
|
else
|
|
{
|
|
prev = column_prev (view, column);
|
|
|
|
if (!prev)
|
|
moo_icon_view_select_path (view, path);
|
|
else
|
|
move_cursor_to_entry (view, prev, prev->entries->len - 1,
|
|
extend_selection);
|
|
}
|
|
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
|
|
static void move_cursor_down (MooIconView *view,
|
|
gboolean extend_selection)
|
|
{
|
|
GtkTreePath *path;
|
|
Column *column, *next;
|
|
int y;
|
|
|
|
path = ensure_cursor (view);
|
|
|
|
column = find_column_by_path (view, path, &y);
|
|
g_return_if_fail (column != NULL);
|
|
|
|
if (y < num_entries (column) - 1)
|
|
{
|
|
move_cursor_to_entry (view, column, y + 1,
|
|
extend_selection);
|
|
}
|
|
else
|
|
{
|
|
next = column_next (view, column);
|
|
|
|
if (!next)
|
|
moo_icon_view_select_path (view, path);
|
|
else
|
|
move_cursor_to_entry (view, next, 0,
|
|
extend_selection);
|
|
}
|
|
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
|
|
static void move_cursor_page_up (MooIconView *view,
|
|
gboolean extend_selection)
|
|
{
|
|
GtkTreePath *path;
|
|
Column *column, *prev;
|
|
int index_;
|
|
|
|
path = ensure_cursor (view);
|
|
|
|
column = find_column_by_path (view, path, &index_);
|
|
g_return_if_fail (column != NULL);
|
|
|
|
prev = column_prev (view, column);
|
|
|
|
if (index_ || !prev)
|
|
move_cursor_to_entry (view, column, 0,
|
|
extend_selection);
|
|
else
|
|
move_cursor_to_entry (view, prev, 0,
|
|
extend_selection);
|
|
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
|
|
static void move_cursor_page_down (MooIconView *view,
|
|
gboolean extend_selection)
|
|
{
|
|
GtkTreePath *path;
|
|
Column *column, *next;
|
|
int index_;
|
|
|
|
path = ensure_cursor (view);
|
|
|
|
column = find_column_by_path (view, path, &index_);
|
|
g_return_if_fail (column != NULL);
|
|
|
|
next = column_next (view, column);
|
|
|
|
if (index_ < num_entries (column) - 1 || !next)
|
|
move_cursor_to_entry (view, column,
|
|
num_entries (column) - 1,
|
|
extend_selection);
|
|
else
|
|
move_cursor_to_entry (view, next,
|
|
num_entries (next) - 1,
|
|
extend_selection);
|
|
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
|
|
static void move_cursor_home (MooIconView *view,
|
|
gboolean extend_selection)
|
|
{
|
|
Column *column;
|
|
column = reinterpret_cast<Column*> (view->priv->layout->columns->data);
|
|
move_cursor_to_entry (view, column, 0,
|
|
extend_selection);
|
|
}
|
|
|
|
|
|
static void move_cursor_end (MooIconView *view,
|
|
gboolean extend_selection)
|
|
{
|
|
Column *column;
|
|
column = reinterpret_cast<Column*> (g_slist_last (view->priv->layout->columns)->data);
|
|
move_cursor_to_entry (view, column,
|
|
num_entries (column) - 1,
|
|
extend_selection);
|
|
}
|
|
|
|
|
|
static void moo_icon_view_scroll_to (MooIconView *view,
|
|
int offset)
|
|
{
|
|
g_return_if_fail (GTK_WIDGET_REALIZED (view));
|
|
|
|
offset = clamp_offset (view, offset);
|
|
|
|
if (offset == view->priv->xoffset)
|
|
return;
|
|
|
|
gdk_window_scroll (GTK_WIDGET(view)->window, view->priv->xoffset - offset, 0);
|
|
view->priv->xoffset = offset;
|
|
|
|
moo_icon_view_update_adjustment (view);
|
|
}
|
|
|
|
|
|
/* this is what GtkRange does */
|
|
static double get_wheel_delta (MooIconView *view)
|
|
{
|
|
GtkAdjustment *adj = view->priv->adjustment;
|
|
|
|
#if 1
|
|
return pow (adj->page_size, 2.0 / 3.0);
|
|
#else
|
|
return adj->step_increment * 2;
|
|
#endif
|
|
}
|
|
|
|
|
|
static gboolean moo_icon_view_scroll_event (GtkWidget *widget,
|
|
GdkEventScroll *event)
|
|
{
|
|
MooIconView *view = MOO_ICON_VIEW (widget);
|
|
int offset = view->priv->xoffset;
|
|
|
|
switch (event->direction)
|
|
{
|
|
case GDK_SCROLL_UP:
|
|
case GDK_SCROLL_LEFT:
|
|
offset -= get_wheel_delta (view);
|
|
break;
|
|
|
|
case GDK_SCROLL_DOWN:
|
|
case GDK_SCROLL_RIGHT:
|
|
offset += get_wheel_delta (view);
|
|
break;
|
|
}
|
|
|
|
moo_icon_view_scroll_to (view, offset);
|
|
return TRUE;
|
|
}
|
|
|
|
|
|
static int clamp_offset (MooIconView *view,
|
|
int offset)
|
|
{
|
|
int layout_width = view->priv->layout->width;
|
|
int width = GTK_WIDGET(view)->allocation.width;
|
|
|
|
if (layout_width <= width)
|
|
return 0;
|
|
else
|
|
return CLAMP (offset, 0, layout_width - width - 1);
|
|
}
|
|
|
|
|
|
/********************************************************************/
|
|
/* _moo_icon_view_get_path_at_pos
|
|
*/
|
|
|
|
static Column *get_column_at_x (MooIconView *view,
|
|
int x)
|
|
{
|
|
GSList *link;
|
|
|
|
if (x < 0 || x >= view->priv->layout->width)
|
|
return NULL;
|
|
|
|
for (link = view->priv->layout->columns; link != NULL; link = link->next)
|
|
{
|
|
Column *column = reinterpret_cast<Column*> (link->data);
|
|
if (column->offset <= x && x < column->offset + column->width)
|
|
return column;
|
|
}
|
|
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static gboolean
|
|
column_get_path_at_xy (MooIconView *view,
|
|
Column *column,
|
|
int x,
|
|
int y,
|
|
GtkTreePath **pathp,
|
|
MooIconViewCell *cell,
|
|
int *cell_x,
|
|
int *cell_y)
|
|
{
|
|
int index_;
|
|
int pixbuf_width;
|
|
GtkTreeIter iter;
|
|
GtkTreePath *path;
|
|
|
|
g_return_val_if_fail (column != NULL, FALSE);
|
|
g_return_val_if_fail (x < column->width, FALSE);
|
|
|
|
index_ = y / view->priv->layout->row_height;
|
|
|
|
if (index_ < 0 || index_ >= num_entries (column))
|
|
return FALSE;
|
|
|
|
path = gtk_tree_path_new_from_indices (index_ + path_get_index (column->first), -1);
|
|
|
|
if (!gtk_tree_model_get_iter (view->priv->model, &iter, path))
|
|
{
|
|
gtk_tree_path_free (path);
|
|
g_return_val_if_reached (FALSE);
|
|
}
|
|
|
|
pixbuf_width = view->priv->layout->pixbuf_width;
|
|
|
|
if (x < pixbuf_width)
|
|
{
|
|
if (pathp)
|
|
*pathp = path;
|
|
else
|
|
gtk_tree_path_free (path);
|
|
if (cell)
|
|
*cell = MOO_ICON_VIEW_CELL_PIXBUF;
|
|
if (cell_x)
|
|
*cell_x = x;
|
|
if (cell_y)
|
|
*cell_y = y - index_ * view->priv->layout->row_height;
|
|
return TRUE;
|
|
}
|
|
else if (view->priv->layout->text_height > 0)
|
|
{
|
|
int text_width;
|
|
|
|
view->priv->text.func (view, view->priv->text.cell,
|
|
view->priv->model, &iter,
|
|
view->priv->text.func_data);
|
|
gtk_cell_renderer_get_size (view->priv->text.cell,
|
|
GTK_WIDGET (view), NULL,
|
|
NULL, NULL, &text_width, NULL);
|
|
|
|
if (x < text_width + pixbuf_width)
|
|
{
|
|
if (pathp)
|
|
*pathp = path;
|
|
else
|
|
gtk_tree_path_free (path);
|
|
|
|
if (cell)
|
|
*cell = MOO_ICON_VIEW_CELL_TEXT;
|
|
|
|
if (cell_x)
|
|
*cell_x = x - pixbuf_width;
|
|
|
|
if (cell_y)
|
|
*cell_y = y - index_ * view->priv->layout->row_height;
|
|
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
gtk_tree_path_free (path);
|
|
return FALSE;
|
|
}
|
|
|
|
static GtkTreePath *
|
|
get_cell_rect (MooIconView *view,
|
|
Column *column,
|
|
int index,
|
|
GdkRectangle *rect)
|
|
{
|
|
GtkTreePath *path;
|
|
GtkTreeIter iter;
|
|
int pixbuf_width;
|
|
int text_width;
|
|
|
|
path = gtk_tree_path_new_from_indices (index + path_get_index (column->first), -1);
|
|
|
|
if (!gtk_tree_model_get_iter (view->priv->model, &iter, path))
|
|
g_return_val_if_reached (NULL);
|
|
|
|
pixbuf_width = view->priv->layout->pixbuf_width;
|
|
text_width = 0;
|
|
|
|
if (view->priv->layout->text_height > 0)
|
|
{
|
|
view->priv->text.func (view, view->priv->text.cell,
|
|
view->priv->model, &iter,
|
|
view->priv->text.func_data);
|
|
gtk_cell_renderer_get_size (view->priv->text.cell,
|
|
GTK_WIDGET (view), NULL,
|
|
NULL, NULL, &text_width, NULL);
|
|
}
|
|
|
|
rect->x = column->offset;
|
|
rect->width = pixbuf_width + text_width;
|
|
rect->y = index * view->priv->layout->row_height;
|
|
rect->height = view->priv->layout->row_height;
|
|
|
|
return path;
|
|
}
|
|
|
|
static GList *
|
|
column_get_paths_in_rect (MooIconView *view,
|
|
Column *column,
|
|
GdkRectangle *rect)
|
|
{
|
|
GList *list = NULL;
|
|
int start, end;
|
|
int i, n;
|
|
|
|
start = rect->y / view->priv->layout->row_height;
|
|
end = (rect->y + rect->height - 1) / view->priv->layout->row_height;
|
|
n = num_entries (column);
|
|
|
|
if (n <= 0 || end < 0 || start >= n)
|
|
return list;
|
|
|
|
start = CLAMP (start, 0, n - 1);
|
|
end = CLAMP (end, 0, n - 1);
|
|
g_return_val_if_fail (start <= end, NULL);
|
|
|
|
for (i = start; i <= end; ++i)
|
|
{
|
|
GtkTreePath *path;
|
|
GdkRectangle cell_rect;
|
|
|
|
if ((path = get_cell_rect (view, column, i, &cell_rect)))
|
|
{
|
|
if (cell_rect.x + cell_rect.width > rect->x &&
|
|
rect->x + rect->width > cell_rect.x)
|
|
list = g_list_prepend (list, path);
|
|
else
|
|
gtk_tree_path_free (path);
|
|
}
|
|
}
|
|
|
|
return g_list_reverse (list);
|
|
}
|
|
|
|
|
|
gboolean
|
|
_moo_icon_view_get_path_at_pos (MooIconView *view,
|
|
int x,
|
|
int y,
|
|
GtkTreePath **path,
|
|
MooIconViewCell *cell,
|
|
int *cell_x,
|
|
int *cell_y)
|
|
{
|
|
Column *column;
|
|
|
|
g_return_val_if_fail (MOO_IS_ICON_VIEW (view), FALSE);
|
|
|
|
column = get_column_at_x (view, x + view->priv->xoffset);
|
|
|
|
if (!column)
|
|
return FALSE;
|
|
else
|
|
return column_get_path_at_xy (view, column,
|
|
x + view->priv->xoffset - column->offset,
|
|
y, path, cell, cell_x, cell_y);
|
|
}
|
|
|
|
|
|
static GList *
|
|
moo_icon_view_get_paths_in_rect (MooIconView *view,
|
|
GdkRectangle *rect)
|
|
{
|
|
GSList *l;
|
|
GList *list = NULL;
|
|
GdkRectangle real_rect;
|
|
|
|
real_rect = *rect;
|
|
real_rect.x += view->priv->xoffset;
|
|
|
|
for (l = view->priv->layout->columns; l != NULL; l = l->next)
|
|
{
|
|
Column *column = reinterpret_cast<Column*> (l->data);
|
|
|
|
if (column->offset + column->width <= real_rect.x)
|
|
continue;
|
|
if (column->offset >= real_rect.x + real_rect.width)
|
|
break;
|
|
|
|
list = g_list_concat (column_get_paths_in_rect (view, column, &real_rect), list);
|
|
}
|
|
|
|
return list;
|
|
}
|
|
|
|
|
|
void
|
|
_moo_icon_view_widget_to_abs_coords (MooIconView *view,
|
|
int wx,
|
|
int wy,
|
|
int *absx,
|
|
int *absy)
|
|
{
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
|
|
if (absx)
|
|
*absx = wx + view->priv->xoffset;
|
|
|
|
if (absy)
|
|
*absy = wy;
|
|
}
|
|
|
|
|
|
void
|
|
_moo_icon_view_abs_to_widget_coords (MooIconView *view,
|
|
int absx,
|
|
int absy,
|
|
int *wx,
|
|
int *wy)
|
|
{
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
|
|
if (wx)
|
|
*wx = absx - view->priv->xoffset;
|
|
|
|
if (wy)
|
|
*wy = absy;
|
|
}
|
|
|
|
|
|
/**************************************************************************/
|
|
/* Selection and cursor
|
|
*/
|
|
|
|
struct Selection {
|
|
GtkSelectionMode mode;
|
|
GSList *selected; /* GtkTreeRowReference* */
|
|
};
|
|
|
|
|
|
static void
|
|
init_selection (MooIconView *view)
|
|
{
|
|
view->priv->selection = g_new (Selection, 1);
|
|
view->priv->selection->mode = GTK_SELECTION_SINGLE;
|
|
view->priv->selection->selected = NULL;
|
|
}
|
|
|
|
|
|
static void
|
|
free_selection (MooIconView *view)
|
|
{
|
|
g_slist_foreach (view->priv->selection->selected,
|
|
(GFunc) gtk_tree_row_reference_free, NULL);
|
|
g_slist_free (view->priv->selection->selected);
|
|
g_free (view->priv->selection);
|
|
view->priv->selection = NULL;
|
|
}
|
|
|
|
|
|
static void
|
|
selection_changed (MooIconView *view)
|
|
{
|
|
g_signal_emit (view, signals[SELECTION_CHANGED], 0);
|
|
}
|
|
|
|
static void
|
|
cursor_moved (MooIconView *view)
|
|
{
|
|
g_signal_emit (view, signals[CURSOR_MOVED], 0);
|
|
}
|
|
|
|
|
|
static void
|
|
selection_clear (MooIconView *view)
|
|
{
|
|
if (view->priv->selection && view->priv->selection->selected)
|
|
{
|
|
g_slist_foreach (view->priv->selection->selected,
|
|
(GFunc) gtk_tree_row_reference_free, NULL);
|
|
g_slist_free (view->priv->selection->selected);
|
|
view->priv->selection->selected = NULL;
|
|
selection_changed (view);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
selection_row_deleted (MooIconView *view)
|
|
{
|
|
GSList *link;
|
|
Selection *sel = view->priv->selection;
|
|
|
|
for (link = sel->selected; link != NULL; link = link->next)
|
|
{
|
|
if (!gtk_tree_row_reference_valid (reinterpret_cast<GtkTreeRowReference*> (link->data)))
|
|
{
|
|
gtk_tree_row_reference_free (reinterpret_cast<GtkTreeRowReference*> (link->data));
|
|
sel->selected = g_slist_delete_link (sel->selected, link);
|
|
selection_changed (view);
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
_moo_icon_view_set_selection_mode (MooIconView *view,
|
|
GtkSelectionMode mode)
|
|
{
|
|
Selection *selection;
|
|
GtkTreePath *path = NULL;
|
|
gboolean select_cursor = FALSE;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
|
|
selection = view->priv->selection;
|
|
|
|
if (selection->mode == mode)
|
|
return;
|
|
|
|
selection->mode = mode;
|
|
|
|
if (!view->priv->model || model_empty (view->priv->model))
|
|
return;
|
|
|
|
if (mode == GTK_SELECTION_MULTIPLE)
|
|
return;
|
|
|
|
switch (mode)
|
|
{
|
|
case GTK_SELECTION_NONE:
|
|
_moo_icon_view_unselect_all (view);
|
|
return;
|
|
|
|
case GTK_SELECTION_BROWSE:
|
|
case GTK_SELECTION_SINGLE:
|
|
break;
|
|
|
|
default:
|
|
g_return_if_reached ();
|
|
}
|
|
|
|
if (mode == GTK_SELECTION_BROWSE)
|
|
select_cursor = TRUE;
|
|
|
|
if (!select_cursor &&
|
|
(path = _moo_icon_view_get_cursor (view)))
|
|
{
|
|
if (_moo_icon_view_path_is_selected (view, path))
|
|
select_cursor = TRUE;
|
|
}
|
|
|
|
_moo_icon_view_unselect_all (view);
|
|
|
|
if (select_cursor)
|
|
{
|
|
if (!path)
|
|
path = _moo_icon_view_get_cursor (view);
|
|
|
|
if (!path)
|
|
{
|
|
path = gtk_tree_path_new_from_indices (0, -1);
|
|
_moo_icon_view_set_cursor (view, path, FALSE);
|
|
}
|
|
else
|
|
{
|
|
moo_icon_view_select_path (view, path);
|
|
}
|
|
}
|
|
|
|
gtk_tree_path_free (path);
|
|
selection_changed (view);
|
|
}
|
|
|
|
|
|
GtkTreePath *
|
|
_moo_icon_view_get_selected_path (MooIconView *view)
|
|
{
|
|
Selection *selection;
|
|
|
|
g_return_val_if_fail (MOO_IS_ICON_VIEW (view), NULL);
|
|
|
|
selection = view->priv->selection;
|
|
|
|
if (!selection->selected)
|
|
return NULL;
|
|
|
|
return gtk_tree_row_reference_get_path (
|
|
reinterpret_cast<GtkTreeRowReference*> (view->priv->selection->selected->data));
|
|
}
|
|
|
|
|
|
gboolean
|
|
_moo_icon_view_get_selected (MooIconView *view,
|
|
GtkTreeIter *iter)
|
|
{
|
|
GtkTreePath *path = NULL;
|
|
|
|
g_return_val_if_fail (MOO_IS_ICON_VIEW (view), FALSE);
|
|
|
|
path = _moo_icon_view_get_selected_path (view);
|
|
|
|
if (path)
|
|
{
|
|
if (iter)
|
|
gtk_tree_model_get_iter (view->priv->model, iter, path);
|
|
gtk_tree_path_free (path);
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
return FALSE;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
_moo_icon_view_selected_foreach (MooIconView *view,
|
|
GtkTreeSelectionForeachFunc func,
|
|
gpointer data)
|
|
{
|
|
Selection *selection;
|
|
GSList *l, *selected;
|
|
GtkTreePath *path;
|
|
GtkTreeIter iter;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (func != NULL);
|
|
|
|
g_object_ref (view);
|
|
|
|
selection = view->priv->selection;
|
|
|
|
for (l = selection->selected, selected = NULL; l != NULL; l = l->next)
|
|
selected = g_slist_prepend (selected, gtk_tree_row_reference_copy (reinterpret_cast<GtkTreeRowReference*> (l->data)));
|
|
selected = g_slist_reverse (selected);
|
|
|
|
while (selected)
|
|
{
|
|
if (gtk_tree_row_reference_valid (reinterpret_cast<GtkTreeRowReference*> (selected->data)))
|
|
{
|
|
path = gtk_tree_row_reference_get_path (reinterpret_cast<GtkTreeRowReference*> (selected->data));
|
|
gtk_tree_model_get_iter (view->priv->model, &iter, path);
|
|
func (view->priv->model, path, &iter, data);
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
gtk_tree_row_reference_free (reinterpret_cast<GtkTreeRowReference*> (selected->data));
|
|
selected = g_slist_delete_link (selected, selected);
|
|
}
|
|
|
|
g_object_unref (view);
|
|
}
|
|
|
|
|
|
static void
|
|
prepend_path (G_GNUC_UNUSED GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
G_GNUC_UNUSED GtkTreeIter *iter,
|
|
GList **list)
|
|
{
|
|
*list = g_list_prepend (*list, gtk_tree_path_copy (path));
|
|
}
|
|
|
|
GList*
|
|
_moo_icon_view_get_selected_rows (MooIconView *view)
|
|
{
|
|
GList *list = NULL;
|
|
_moo_icon_view_selected_foreach (view,
|
|
(GtkTreeSelectionForeachFunc) prepend_path,
|
|
&list);
|
|
return g_list_reverse (list);
|
|
}
|
|
|
|
|
|
#if 0
|
|
int
|
|
_moo_icon_view_count_selected_rows (MooIconView *view)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_ICON_VIEW (view), 0);
|
|
return g_slist_length (view->priv->selection->selected);
|
|
}
|
|
#endif
|
|
|
|
|
|
static int
|
|
row_reference_compare (GtkTreeRowReference *ref1,
|
|
GtkTreeRowReference *ref2)
|
|
{
|
|
int result;
|
|
GtkTreePath *path1, *path2;
|
|
path1 = gtk_tree_row_reference_get_path (ref1);
|
|
path2 = gtk_tree_row_reference_get_path (ref2);
|
|
result = gtk_tree_path_compare (path1, path2);
|
|
gtk_tree_path_free (path1);
|
|
gtk_tree_path_free (path2);
|
|
return result;
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_select_path (MooIconView *view,
|
|
GtkTreePath *path)
|
|
{
|
|
Selection *selection;
|
|
GtkTreeRowReference *ref;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (path != NULL);
|
|
g_return_if_fail (gtk_tree_path_get_depth (path) == 1);
|
|
|
|
selection = view->priv->selection;
|
|
g_return_if_fail (selection->mode != GTK_SELECTION_NONE);
|
|
|
|
ref = gtk_tree_row_reference_new (view->priv->model, path);
|
|
g_return_if_fail (ref != NULL);
|
|
|
|
if (selection->mode == GTK_SELECTION_SINGLE ||
|
|
selection->mode == GTK_SELECTION_BROWSE)
|
|
{
|
|
_moo_icon_view_unselect_all (view);
|
|
g_assert (selection->selected == NULL);
|
|
selection->selected = g_slist_prepend (selection->selected, ref);
|
|
invalidate_path_rectangle (view, path);
|
|
selection_changed (view);
|
|
return;
|
|
}
|
|
|
|
if (!g_slist_find_custom (view->priv->selection->selected, ref,
|
|
(GCompareFunc) row_reference_compare))
|
|
{
|
|
selection->selected = g_slist_prepend (selection->selected, ref);
|
|
invalidate_path_rectangle (view, path);
|
|
selection_changed (view);
|
|
return;
|
|
}
|
|
}
|
|
|
|
static void
|
|
moo_icon_view_select_paths (MooIconView *view,
|
|
GList *list)
|
|
{
|
|
/* XXX */
|
|
while (list)
|
|
{
|
|
moo_icon_view_select_path (view, reinterpret_cast<GtkTreePath*> (list->data));
|
|
list = list->next;
|
|
}
|
|
}
|
|
|
|
|
|
void
|
|
_moo_icon_view_select_all (MooIconView *view)
|
|
{
|
|
GtkTreeIter iter;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
|
|
if (view->priv->model && gtk_tree_model_get_iter_first (view->priv->model, &iter))
|
|
{
|
|
int n_children;
|
|
GtkTreePath *start, *end;
|
|
|
|
n_children = gtk_tree_model_iter_n_children (view->priv->model, NULL);
|
|
start = gtk_tree_model_get_path (view->priv->model, &iter);
|
|
gtk_tree_model_iter_nth_child (view->priv->model, &iter, NULL, n_children - 1);
|
|
end = gtk_tree_model_get_path (view->priv->model, &iter);
|
|
|
|
moo_icon_view_select_range (view, start, end);
|
|
|
|
gtk_tree_path_free (end);
|
|
gtk_tree_path_free (start);
|
|
}
|
|
}
|
|
|
|
static void
|
|
moo_icon_view_select_range (MooIconView *view,
|
|
GtkTreePath *start,
|
|
GtkTreePath *end)
|
|
{
|
|
Selection *selection;
|
|
GtkTreePath *path;
|
|
int start_index, end_index, i;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (start != NULL && end != NULL);
|
|
g_return_if_fail (gtk_tree_path_get_depth (start) == 1);
|
|
g_return_if_fail (gtk_tree_path_get_depth (end) == 1);
|
|
|
|
selection = view->priv->selection;
|
|
g_return_if_fail (selection->mode != GTK_SELECTION_NONE);
|
|
|
|
if (selection->mode != GTK_SELECTION_MULTIPLE)
|
|
{
|
|
g_return_if_fail (!gtk_tree_path_compare (start, end));
|
|
moo_icon_view_select_path (view, start);
|
|
return;
|
|
}
|
|
|
|
start_index = gtk_tree_path_get_indices(start)[0];
|
|
end_index = gtk_tree_path_get_indices(end)[0];
|
|
|
|
if (start_index > end_index)
|
|
{
|
|
int tmp = start_index;
|
|
start_index = end_index;
|
|
end_index = tmp;
|
|
}
|
|
|
|
path = gtk_tree_path_new_from_indices (start_index, -1);
|
|
|
|
for (i = start_index; i <= end_index; ++i, gtk_tree_path_next (path))
|
|
moo_icon_view_select_path (view, path);
|
|
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_unselect_path (MooIconView *view,
|
|
GtkTreePath *path)
|
|
{
|
|
Selection *selection;
|
|
GSList *link;
|
|
GtkTreePath *selected;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (path != NULL);
|
|
|
|
if (gtk_tree_path_get_depth (path) != 1)
|
|
return;
|
|
|
|
selection = view->priv->selection;
|
|
|
|
for (link = selection->selected; link != NULL; link = link->next)
|
|
{
|
|
g_assert (gtk_tree_row_reference_valid (reinterpret_cast<GtkTreeRowReference*> (link->data)));
|
|
selected = gtk_tree_row_reference_get_path (reinterpret_cast<GtkTreeRowReference*> (link->data));
|
|
|
|
if (!gtk_tree_path_compare (selected, path))
|
|
{
|
|
gtk_tree_row_reference_free (reinterpret_cast<GtkTreeRowReference*> (link->data));
|
|
gtk_tree_path_free (selected);
|
|
selection->selected =
|
|
g_slist_delete_link (selection->selected, link);
|
|
invalidate_path_rectangle (view, path);
|
|
selection_changed (view);
|
|
return;
|
|
}
|
|
|
|
gtk_tree_path_free (selected);
|
|
}
|
|
}
|
|
|
|
|
|
gboolean
|
|
_moo_icon_view_path_is_selected (MooIconView *view,
|
|
GtkTreePath *path)
|
|
{
|
|
Selection *selection;
|
|
GSList *link;
|
|
GtkTreePath *selected;
|
|
|
|
g_return_val_if_fail (MOO_IS_ICON_VIEW (view), FALSE);
|
|
g_return_val_if_fail (path != NULL, FALSE);
|
|
|
|
if (gtk_tree_path_get_depth (path) != 1)
|
|
return FALSE;
|
|
|
|
selection = view->priv->selection;
|
|
|
|
for (link = selection->selected; link != NULL; link = link->next)
|
|
{
|
|
g_assert (gtk_tree_row_reference_valid (reinterpret_cast<GtkTreeRowReference*> (link->data)));
|
|
selected = gtk_tree_row_reference_get_path (reinterpret_cast<GtkTreeRowReference*> (link->data));
|
|
|
|
if (!gtk_tree_path_compare (selected, path))
|
|
{
|
|
gtk_tree_path_free (selected);
|
|
return TRUE;
|
|
}
|
|
|
|
gtk_tree_path_free (selected);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
#if 0
|
|
void
|
|
moo_icon_view_select_all (MooIconView *view)
|
|
{
|
|
GtkTreeIter iter;
|
|
Selection *selection;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
|
|
if (!view->priv->model || model_empty (view->priv->model))
|
|
return;
|
|
|
|
selection = view->priv->selection;
|
|
g_slist_foreach (selection->selected,
|
|
(GFunc) gtk_tree_row_reference_free, NULL);
|
|
g_slist_free (selection->selected);
|
|
selection->selected = NULL;
|
|
|
|
gtk_tree_model_get_iter_first (view->priv->model, &iter);
|
|
|
|
do {
|
|
GtkTreePath *path;
|
|
GtkTreeRowReference *ref;
|
|
path = gtk_tree_model_get_path (view->priv->model, &iter);
|
|
g_return_if_fail (path != NULL);
|
|
ref = gtk_tree_row_reference_new (view->priv->model, path);
|
|
selection->selected = g_slist_prepend (selection->selected, ref);
|
|
gtk_tree_path_free (path);
|
|
}
|
|
while (gtk_tree_model_iter_next (view->priv->model, &iter));
|
|
|
|
selection->selected = g_slist_reverse (selection->selected);
|
|
gtk_widget_queue_draw (GTK_WIDGET (view));
|
|
selection_changed (view);
|
|
}
|
|
#endif
|
|
|
|
|
|
void
|
|
_moo_icon_view_unselect_all (MooIconView *view)
|
|
{
|
|
Selection *selection;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
|
|
selection = view->priv->selection;
|
|
|
|
if (!selection->selected)
|
|
return;
|
|
|
|
g_return_if_fail (selection->mode != GTK_SELECTION_BROWSE);
|
|
|
|
g_slist_foreach (selection->selected,
|
|
(GFunc) gtk_tree_row_reference_free, NULL);
|
|
g_slist_free (selection->selected);
|
|
selection->selected = NULL;
|
|
|
|
gtk_widget_queue_draw (GTK_WIDGET (view));
|
|
selection_changed (view);
|
|
}
|
|
|
|
|
|
void
|
|
_moo_icon_view_scroll_to_cell (MooIconView *view,
|
|
GtkTreePath *path)
|
|
{
|
|
Column *column;
|
|
int xoffset, new_offset;
|
|
GtkWidget *widget;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (path != NULL);
|
|
g_return_if_fail (gtk_tree_path_get_depth (path) == 1);
|
|
g_return_if_fail (view->priv->model != NULL);
|
|
g_return_if_fail (!model_empty (view->priv->model));
|
|
|
|
if (!GTK_WIDGET_REALIZED (view) || view->priv->update_idle)
|
|
{
|
|
if (view->priv->scroll_to)
|
|
gtk_tree_row_reference_free (view->priv->scroll_to);
|
|
view->priv->scroll_to = gtk_tree_row_reference_new (view->priv->model, path);
|
|
return;
|
|
}
|
|
|
|
column = find_column_by_path (view, path, NULL);
|
|
g_return_if_fail (column != NULL);
|
|
|
|
xoffset = view->priv->xoffset;
|
|
new_offset = xoffset;
|
|
widget = GTK_WIDGET(view);
|
|
|
|
if (widget->allocation.width <= column->width)
|
|
new_offset = column->offset;
|
|
else if (column->offset < xoffset)
|
|
new_offset = column->offset;
|
|
else if (column->offset + column->width > xoffset + widget->allocation.width)
|
|
new_offset = column->offset + column->width - widget->allocation.width;
|
|
|
|
moo_icon_view_scroll_to (view, new_offset);
|
|
}
|
|
|
|
|
|
void
|
|
_moo_icon_view_set_cursor (MooIconView *view,
|
|
GtkTreePath *path,
|
|
gboolean start_editing)
|
|
{
|
|
GtkTreeRowReference *ref;
|
|
GtkTreePath *old;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (path != NULL);
|
|
|
|
ref = gtk_tree_row_reference_new (view->priv->model, path);
|
|
g_return_if_fail (ref != NULL);
|
|
|
|
old = _moo_icon_view_get_cursor (view);
|
|
|
|
if (old)
|
|
{
|
|
invalidate_path_rectangle (view, old);
|
|
gtk_tree_path_free (old);
|
|
gtk_tree_row_reference_free (view->priv->cursor);
|
|
}
|
|
|
|
view->priv->cursor = ref;
|
|
moo_icon_view_select_path (view, path);
|
|
cursor_moved (view);
|
|
|
|
if (start_editing)
|
|
{
|
|
_moo_icon_view_scroll_to_cell (view, path);
|
|
/* TODO */
|
|
g_warning ("implement me");
|
|
}
|
|
}
|
|
|
|
|
|
static GtkTreePath *
|
|
_moo_icon_view_get_cursor (MooIconView *view)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_ICON_VIEW (view), NULL);
|
|
|
|
if (view->priv->cursor)
|
|
return gtk_tree_row_reference_get_path (view->priv->cursor);
|
|
else
|
|
return NULL;
|
|
}
|
|
|
|
|
|
static void
|
|
cursor_row_deleted (MooIconView *view)
|
|
{
|
|
if (view->priv->cursor)
|
|
{
|
|
if (!gtk_tree_row_reference_valid (view->priv->cursor))
|
|
{
|
|
gtk_tree_row_reference_free (view->priv->cursor);
|
|
view->priv->cursor = NULL;
|
|
cursor_moved (view);
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
_moo_icon_view_row_activated (MooIconView *view,
|
|
GtkTreePath *path)
|
|
{
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
g_return_if_fail (path != NULL);
|
|
g_signal_emit (view, signals[ROW_ACTIVATED], 0, path);
|
|
}
|
|
|
|
|
|
static void
|
|
activate_item_at_cursor (MooIconView *view)
|
|
{
|
|
GtkTreePath *path;
|
|
|
|
if (!view->priv->cursor)
|
|
return;
|
|
|
|
path = gtk_tree_row_reference_get_path (view->priv->cursor);
|
|
g_return_if_fail (path != NULL);
|
|
_moo_icon_view_row_activated (view, path);
|
|
gtk_tree_path_free (path);
|
|
}
|
|
|
|
|
|
/********************************************************************/
|
|
/* Drag'n'drop
|
|
*/
|
|
|
|
void
|
|
_moo_icon_view_enable_drag_source (MooIconView *view,
|
|
GdkModifierType start_button_mask,
|
|
GtkTargetEntry *targets,
|
|
gint n_targets,
|
|
GdkDragAction actions)
|
|
{
|
|
DndInfo *info;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
|
|
gtk_drag_source_set (GTK_WIDGET (view), GdkModifierType (0), targets, n_targets, actions);
|
|
|
|
info = view->priv->dnd_info;
|
|
|
|
if (info->source_targets)
|
|
gtk_target_list_unref (info->source_targets);
|
|
|
|
info->start_button_mask = start_button_mask;
|
|
info->source_targets = gtk_target_list_new (targets, n_targets);
|
|
info->source_actions = actions;
|
|
info->source_enabled = TRUE;
|
|
}
|
|
|
|
|
|
#if 0
|
|
GtkTargetList*
|
|
_moo_icon_view_get_source_targets (MooIconView *view)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_ICON_VIEW (view), NULL);
|
|
return view->priv->dnd_info->source_targets;
|
|
}
|
|
|
|
void
|
|
_moo_icon_view_disable_drag_source (MooIconView *view)
|
|
{
|
|
DndInfo *info;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
|
|
info = view->priv->dnd_info;
|
|
|
|
if (info->source_enabled)
|
|
{
|
|
gtk_drag_source_unset (GTK_WIDGET (view));
|
|
gtk_target_list_unref (info->source_targets);
|
|
info->source_targets = NULL;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
static gboolean
|
|
moo_icon_view_maybe_drag (MooIconView *view,
|
|
GdkEventMotion *event)
|
|
{
|
|
DndInfo *info;
|
|
int button;
|
|
|
|
info = view->priv->dnd_info;
|
|
|
|
if (!view->priv->button_pressed)
|
|
return FALSE;
|
|
|
|
if (!info->source_enabled)
|
|
return FALSE;
|
|
|
|
if (!gtk_drag_check_threshold (GTK_WIDGET (view),
|
|
view->priv->button_press_x - view->priv->xoffset,
|
|
view->priv->button_press_y,
|
|
(int) event->x, (int) event->y))
|
|
return FALSE;
|
|
|
|
button = view->priv->button_pressed;
|
|
|
|
if (!_moo_icon_view_get_path_at_pos (view,
|
|
view->priv->button_press_x - view->priv->xoffset,
|
|
view->priv->button_press_y,
|
|
NULL, NULL, NULL, NULL))
|
|
return FALSE;
|
|
|
|
if (!(GDK_BUTTON1_MASK << (button - 1) & info->start_button_mask))
|
|
return FALSE;
|
|
|
|
if (view->priv->button_press_row)
|
|
{
|
|
gtk_tree_row_reference_free (view->priv->button_press_row);
|
|
view->priv->button_press_row = NULL;
|
|
}
|
|
|
|
gtk_drag_begin (GTK_WIDGET (view),
|
|
info->source_targets,
|
|
info->source_actions,
|
|
button,
|
|
(GdkEvent*) event);
|
|
view->priv->button_pressed = 0;
|
|
return TRUE;
|
|
|
|
#if 0
|
|
// GdkPixmap *pixmap;
|
|
// pixmap = moo_icon_view_create_row_drag_icon (view, path);
|
|
// gtk_drag_set_icon_pixmap (context,
|
|
// gdk_drawable_get_colormap (pixmap),
|
|
// pixmap,
|
|
// NULL,
|
|
// /* the + 1 is for the black border in the icon ? */
|
|
// view->priv->button_press_x + 1,
|
|
// 1);
|
|
#endif
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_drag_begin (G_GNUC_UNUSED GtkWidget *widget,
|
|
G_GNUC_UNUSED GdkDragContext *context)
|
|
{
|
|
}
|
|
|
|
|
|
static void
|
|
moo_icon_view_drag_end (GtkWidget *widget,
|
|
G_GNUC_UNUSED GdkDragContext *context)
|
|
{
|
|
MooIconView *view = MOO_ICON_VIEW (widget);
|
|
view->priv->button_pressed = 0;
|
|
}
|
|
|
|
|
|
void
|
|
_moo_icon_view_enable_drag_dest (MooIconView *view,
|
|
GtkTargetEntry *targets,
|
|
gint n_targets,
|
|
GdkDragAction actions)
|
|
{
|
|
DndInfo *info;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
|
|
gtk_drag_dest_set (GTK_WIDGET (view), GtkDestDefaults (0), targets, n_targets, actions);
|
|
|
|
info = view->priv->dnd_info;
|
|
|
|
if (info->dest_targets)
|
|
gtk_target_list_unref (info->dest_targets);
|
|
|
|
info->dest_targets = gtk_target_list_new (targets, n_targets);
|
|
info->dest_actions = actions;
|
|
info->dest_enabled = TRUE;
|
|
}
|
|
|
|
|
|
void
|
|
_moo_icon_view_set_dest_targets (MooIconView *view,
|
|
GtkTargetList *targets)
|
|
{
|
|
DndInfo *info;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
|
|
info = view->priv->dnd_info;
|
|
g_return_if_fail (info->dest_enabled);
|
|
|
|
if (info->dest_targets)
|
|
gtk_target_list_unref (info->dest_targets);
|
|
|
|
gtk_target_list_ref (targets);
|
|
info->dest_targets = targets;
|
|
gtk_drag_dest_set_target_list (GTK_WIDGET (view), targets);
|
|
}
|
|
|
|
|
|
#if 0
|
|
void
|
|
_moo_icon_view_disable_drag_dest (MooIconView *view)
|
|
{
|
|
DndInfo *info;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
|
|
info = view->priv->dnd_info;
|
|
|
|
if (info->dest_enabled)
|
|
{
|
|
gtk_drag_dest_unset (GTK_WIDGET (view));
|
|
gtk_target_list_unref (info->dest_targets);
|
|
info->dest_targets = NULL;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
|
|
static void
|
|
moo_icon_view_drag_leave (G_GNUC_UNUSED GtkWidget *widget,
|
|
G_GNUC_UNUSED GdkDragContext *context,
|
|
G_GNUC_UNUSED guint time)
|
|
{
|
|
DndInfo *info;
|
|
MooIconView *view = MOO_ICON_VIEW (widget);
|
|
|
|
info = view->priv->dnd_info;
|
|
|
|
if (info->drag_motion_context)
|
|
g_object_unref (info->drag_motion_context);
|
|
info->drag_motion_context = NULL;
|
|
|
|
if (!info->drag_dest_inside)
|
|
g_warning ("drag_leave: oops\n");
|
|
|
|
info->drag_dest_inside = FALSE;
|
|
drag_scroll_stop (view);
|
|
}
|
|
|
|
|
|
static void
|
|
drag_scroll_stop (MooIconView *view)
|
|
{
|
|
if (view->priv->drag_scroll_timeout)
|
|
{
|
|
g_source_remove (view->priv->drag_scroll_timeout);
|
|
view->priv->drag_scroll_timeout = 0;
|
|
}
|
|
}
|
|
|
|
|
|
#define DRAG_SCROLL_MARGIN 0.1
|
|
#define DRAG_SCROLL_TIMEOUT 100
|
|
|
|
static gboolean
|
|
drag_scroll_timeout (MooIconView *view)
|
|
{
|
|
GtkWidget *widget = GTK_WIDGET (view);
|
|
GtkAllocation *alc = &widget->allocation;
|
|
GtkWidget *toplevel;
|
|
int x, y, new_offset;
|
|
int delta, dist;
|
|
double ratio, margin;
|
|
GdkModifierType mask;
|
|
GdkEvent *event;
|
|
DndInfo *info = view->priv->dnd_info;
|
|
|
|
gdk_window_get_pointer (widget->window, &x, &y, &mask);
|
|
|
|
if (view->priv->drag_select)
|
|
{
|
|
if (x < 0)
|
|
delta = x;
|
|
else
|
|
delta = x - widget->allocation.width;
|
|
}
|
|
else
|
|
{
|
|
if (x < 0 || x >= alc->width || y < 0 || y >= alc->height)
|
|
goto out;
|
|
|
|
if (x < widget->allocation.width * DRAG_SCROLL_MARGIN)
|
|
{
|
|
dist = x;
|
|
delta = -1;
|
|
}
|
|
else if (x > widget->allocation.width * (1 - DRAG_SCROLL_MARGIN))
|
|
{
|
|
dist = widget->allocation.width - 1 - x;
|
|
delta = 1;
|
|
}
|
|
else
|
|
{
|
|
goto out;
|
|
}
|
|
|
|
margin = widget->allocation.width * DRAG_SCROLL_MARGIN;
|
|
margin = MAX (margin, 2);
|
|
dist = CLAMP (dist, 1, margin - 1);
|
|
ratio = (margin - dist) / margin;
|
|
delta *= 15 * (1 + 3 * ratio);
|
|
}
|
|
|
|
new_offset = clamp_offset (view, view->priv->xoffset + delta);
|
|
|
|
if (new_offset == view->priv->xoffset)
|
|
goto out;
|
|
|
|
moo_icon_view_scroll_to (view, new_offset);
|
|
|
|
toplevel = gtk_widget_get_toplevel (widget);
|
|
|
|
if (!GTK_IS_WINDOW (toplevel))
|
|
{
|
|
g_critical ("oops");
|
|
goto out;
|
|
}
|
|
|
|
if (view->priv->drag_select)
|
|
{
|
|
event = gdk_event_new (GDK_MOTION_NOTIFY);
|
|
|
|
event->motion.window = GDK_WINDOW (g_object_ref (widget->window));
|
|
event->motion.axes = NULL;
|
|
gdk_window_get_pointer (widget->window, &x, &y, &mask);
|
|
event->motion.x = x;
|
|
event->motion.y = y;
|
|
event->motion.state = mask;
|
|
event->motion.is_hint = FALSE;
|
|
/* XXX ??? do I need it, is it right? */
|
|
event->motion.device = gdk_device_get_core_pointer ();
|
|
gdk_window_get_position (toplevel->window, &x, &y);
|
|
event->motion.x_root = x + event->motion.x;
|
|
event->motion.y_root = y + event->motion.y;
|
|
}
|
|
else
|
|
{
|
|
event = gdk_event_new (GDK_DRAG_MOTION);
|
|
|
|
event->dnd.window = GDK_WINDOW(g_object_ref (toplevel->window));
|
|
event->dnd.context = GDK_DRAG_CONTEXT (g_object_ref (info->drag_motion_context));
|
|
|
|
gdk_window_get_position (toplevel->window, &x, &y);
|
|
event->dnd.x_root = x;
|
|
event->dnd.y_root = y;
|
|
gdk_window_get_pointer (toplevel->window, &x, &y, &mask);
|
|
event->dnd.x_root += x;
|
|
event->dnd.y_root += y;
|
|
}
|
|
|
|
event->any.send_event = TRUE;
|
|
gtk_main_do_event (event);
|
|
gdk_event_free (event);
|
|
|
|
return TRUE;
|
|
|
|
out:
|
|
drag_scroll_stop (view);
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
static void
|
|
drag_scroll_check (MooIconView *view,
|
|
int x,
|
|
int y)
|
|
{
|
|
gboolean need_scroll;
|
|
GtkAllocation *alc = >K_WIDGET(view)->allocation;
|
|
|
|
if (view->priv->drag_select)
|
|
need_scroll = x < 0 || x >= alc->width;
|
|
else
|
|
need_scroll = x >= 0 && x < alc->width && y >= 0 && y < alc->height &&
|
|
(x < alc->width * DRAG_SCROLL_MARGIN ||
|
|
x > alc->width * (1 - DRAG_SCROLL_MARGIN));
|
|
|
|
if (!need_scroll)
|
|
drag_scroll_stop (view);
|
|
else if (!view->priv->drag_scroll_timeout)
|
|
view->priv->drag_scroll_timeout =
|
|
gdk_threads_add_timeout (DRAG_SCROLL_TIMEOUT,
|
|
(GSourceFunc) drag_scroll_timeout,
|
|
view);
|
|
}
|
|
|
|
|
|
static gboolean
|
|
moo_icon_view_drag_motion (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
int x,
|
|
int y,
|
|
G_GNUC_UNUSED guint time)
|
|
{
|
|
DndInfo *info;
|
|
GdkAtom target;
|
|
MooIconView *view = MOO_ICON_VIEW (widget);
|
|
|
|
info = view->priv->dnd_info;
|
|
target = gtk_drag_dest_find_target (widget, context, info->dest_targets);
|
|
|
|
if (target == GDK_NONE)
|
|
{
|
|
if (info->drag_motion_context)
|
|
{
|
|
g_critical ("oops");
|
|
g_object_unref (info->drag_motion_context);
|
|
info->drag_motion_context = NULL;
|
|
}
|
|
|
|
drag_scroll_stop (view);
|
|
return FALSE;
|
|
}
|
|
|
|
if (info->drag_motion_context != context)
|
|
{
|
|
if (info->drag_motion_context)
|
|
{
|
|
g_critical ("oops");
|
|
g_object_unref (info->drag_motion_context);
|
|
}
|
|
|
|
info->drag_motion_context = GDK_DRAG_CONTEXT (g_object_ref (context));
|
|
}
|
|
|
|
if (!info->drag_dest_inside)
|
|
info->drag_dest_inside = TRUE;
|
|
else
|
|
drag_scroll_check (view, x, y);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
|
|
void
|
|
_moo_icon_view_set_drag_dest_row (MooIconView *view,
|
|
GtkTreePath *path)
|
|
{
|
|
GtkTreeRowReference *ref = NULL;
|
|
|
|
g_return_if_fail (MOO_IS_ICON_VIEW (view));
|
|
|
|
if (path)
|
|
ref = gtk_tree_row_reference_new (view->priv->model, path);
|
|
|
|
if (view->priv->drop_dest)
|
|
{
|
|
GtkTreePath *old_path;
|
|
|
|
old_path = gtk_tree_row_reference_get_path (view->priv->drop_dest);
|
|
|
|
if (old_path)
|
|
{
|
|
invalidate_path_rectangle (view, old_path);
|
|
gtk_tree_path_free (old_path);
|
|
}
|
|
|
|
gtk_tree_row_reference_free (view->priv->drop_dest);
|
|
view->priv->drop_dest = NULL;
|
|
}
|
|
|
|
if (ref)
|
|
{
|
|
view->priv->drop_dest = ref;
|
|
invalidate_path_rectangle (view, path);
|
|
}
|
|
}
|
|
|
|
|
|
GtkTreePath *
|
|
_moo_icon_view_get_drag_dest_row (MooIconView *view)
|
|
{
|
|
g_return_val_if_fail (MOO_IS_ICON_VIEW (view), NULL);
|
|
|
|
if (view->priv->drop_dest)
|
|
return gtk_tree_row_reference_get_path (view->priv->drop_dest);
|
|
else
|
|
return NULL;
|
|
}
|