forked from AuroraMiddleware/gtk
f330b40521
Added concept of "Focus Siblings" to GtkCellArea so that some static text/icon may be included in the focus/click area of an activatable or editable cell, implemented focus drawing as well, updated testcellarea to reflect the changes.
1017 lines
30 KiB
C
1017 lines
30 KiB
C
/* cellareascaffold.c
|
|
*
|
|
* Copyright (C) 2010 Openismus GmbH
|
|
*
|
|
* Authors:
|
|
* Tristan Van Berkom <tristanvb@openismus.com>
|
|
*
|
|
* This library is free software; you can redistribute it and/or
|
|
* modify it under the terms of the GNU Library General Public
|
|
* License as published by the Free Software Foundation; either
|
|
* version 2 of the License, or (at your option) any later version.
|
|
*
|
|
* This library is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
* Library General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Library General Public
|
|
* License along with this library; if not, write to the
|
|
* Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
* Boston, MA 02111-1307, USA.
|
|
*/
|
|
|
|
#include <string.h>
|
|
#include "cellareascaffold.h"
|
|
|
|
/* GObjectClass */
|
|
static void cell_area_scaffold_finalize (GObject *object);
|
|
static void cell_area_scaffold_dispose (GObject *object);
|
|
static void cell_area_scaffold_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec);
|
|
static void cell_area_scaffold_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec);
|
|
|
|
/* GtkWidgetClass */
|
|
static void cell_area_scaffold_realize (GtkWidget *widget);
|
|
static void cell_area_scaffold_unrealize (GtkWidget *widget);
|
|
static gboolean cell_area_scaffold_draw (GtkWidget *widget,
|
|
cairo_t *cr);
|
|
static void cell_area_scaffold_size_allocate (GtkWidget *widget,
|
|
GtkAllocation *allocation);
|
|
static void cell_area_scaffold_get_preferred_width (GtkWidget *widget,
|
|
gint *minimum_size,
|
|
gint *natural_size);
|
|
static void cell_area_scaffold_get_preferred_height_for_width (GtkWidget *widget,
|
|
gint for_size,
|
|
gint *minimum_size,
|
|
gint *natural_size);
|
|
static void cell_area_scaffold_get_preferred_height (GtkWidget *widget,
|
|
gint *minimum_size,
|
|
gint *natural_size);
|
|
static void cell_area_scaffold_get_preferred_width_for_height (GtkWidget *widget,
|
|
gint for_size,
|
|
gint *minimum_size,
|
|
gint *natural_size);
|
|
static gint cell_area_scaffold_focus (GtkWidget *widget,
|
|
GtkDirectionType direction);
|
|
|
|
/* CellAreaScaffoldClass */
|
|
static void cell_area_scaffold_activate (CellAreaScaffold *scaffold);
|
|
|
|
/* CellArea/GtkTreeModel callbacks */
|
|
static void size_changed_cb (GtkCellAreaIter *iter,
|
|
GParamSpec *pspec,
|
|
CellAreaScaffold *scaffold);
|
|
static void row_changed_cb (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
GtkTreeIter *iter,
|
|
CellAreaScaffold *scaffold);
|
|
static void row_inserted_cb (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
GtkTreeIter *iter,
|
|
CellAreaScaffold *scaffold);
|
|
static void row_deleted_cb (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
CellAreaScaffold *scaffold);
|
|
static void rows_reordered_cb (GtkTreeModel *model,
|
|
GtkTreePath *parent,
|
|
GtkTreeIter *iter,
|
|
gint *new_order,
|
|
CellAreaScaffold *scaffold);
|
|
|
|
typedef struct {
|
|
gint size; /* The size of the row in the scaffold's opposing orientation */
|
|
} RowData;
|
|
|
|
struct _CellAreaScaffoldPrivate {
|
|
|
|
/* Window for catching events and dispatching them to the cell area */
|
|
GdkWindow *event_window;
|
|
|
|
/* The model we're showing data for */
|
|
GtkTreeModel *model;
|
|
gulong row_changed_id;
|
|
gulong row_inserted_id;
|
|
gulong row_deleted_id;
|
|
gulong rows_reordered_id;
|
|
|
|
/* The area rendering the data and a global iter */
|
|
GtkCellArea *area;
|
|
GtkCellAreaIter *iter;
|
|
|
|
/* Cache some info about rows (hieghts etc) */
|
|
GArray *row_data;
|
|
|
|
/* Focus handling */
|
|
gint focus_row;
|
|
|
|
/* Check when the underlying area changes the size and
|
|
* we need to queue a redraw */
|
|
gulong size_changed_id;
|
|
|
|
|
|
};
|
|
|
|
enum {
|
|
PROP_0,
|
|
PROP_ORIENTATION
|
|
};
|
|
|
|
enum {
|
|
ACTIVATE,
|
|
N_SIGNALS
|
|
};
|
|
|
|
static guint scaffold_signals[N_SIGNALS] = { 0 };
|
|
|
|
#define ROW_SPACING 2
|
|
|
|
#define DIRECTION_STR(dir) \
|
|
((dir) == GTK_DIR_TAB_FORWARD ? "tab forward" : \
|
|
(dir) == GTK_DIR_TAB_BACKWARD ? "tab backward" : \
|
|
(dir) == GTK_DIR_UP ? "up" : \
|
|
(dir) == GTK_DIR_DOWN ? "down" : \
|
|
(dir) == GTK_DIR_LEFT ? "left" : \
|
|
(dir) == GTK_DIR_RIGHT ? "right" : "invalid")
|
|
|
|
G_DEFINE_TYPE_WITH_CODE (CellAreaScaffold, cell_area_scaffold, GTK_TYPE_WIDGET,
|
|
G_IMPLEMENT_INTERFACE (GTK_TYPE_ORIENTABLE, NULL));
|
|
|
|
|
|
static void
|
|
cell_area_scaffold_init (CellAreaScaffold *scaffold)
|
|
{
|
|
CellAreaScaffoldPrivate *priv;
|
|
|
|
scaffold->priv = G_TYPE_INSTANCE_GET_PRIVATE (scaffold,
|
|
TYPE_CELL_AREA_SCAFFOLD,
|
|
CellAreaScaffoldPrivate);
|
|
priv = scaffold->priv;
|
|
|
|
priv->area = gtk_cell_area_box_new ();
|
|
priv->iter = gtk_cell_area_create_iter (priv->area);
|
|
|
|
priv->row_data = g_array_new (FALSE, FALSE, sizeof (RowData));
|
|
|
|
gtk_widget_set_has_window (GTK_WIDGET (scaffold), FALSE);
|
|
gtk_widget_set_can_focus (GTK_WIDGET (scaffold), TRUE);
|
|
|
|
priv->size_changed_id =
|
|
g_signal_connect (priv->iter, "notify",
|
|
G_CALLBACK (size_changed_cb), scaffold);
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_class_init (CellAreaScaffoldClass *class)
|
|
{
|
|
GObjectClass *gobject_class;
|
|
GtkWidgetClass *widget_class;
|
|
|
|
gobject_class = G_OBJECT_CLASS(class);
|
|
gobject_class->dispose = cell_area_scaffold_dispose;
|
|
gobject_class->finalize = cell_area_scaffold_finalize;
|
|
gobject_class->get_property = cell_area_scaffold_get_property;
|
|
gobject_class->set_property = cell_area_scaffold_set_property;
|
|
|
|
widget_class = GTK_WIDGET_CLASS(class);
|
|
widget_class->realize = cell_area_scaffold_realize;
|
|
widget_class->unrealize = cell_area_scaffold_unrealize;
|
|
widget_class->draw = cell_area_scaffold_draw;
|
|
widget_class->size_allocate = cell_area_scaffold_size_allocate;
|
|
widget_class->get_preferred_width = cell_area_scaffold_get_preferred_width;
|
|
widget_class->get_preferred_height_for_width = cell_area_scaffold_get_preferred_height_for_width;
|
|
widget_class->get_preferred_height = cell_area_scaffold_get_preferred_height;
|
|
widget_class->get_preferred_width_for_height = cell_area_scaffold_get_preferred_width_for_height;
|
|
widget_class->focus = cell_area_scaffold_focus;
|
|
|
|
class->activate = cell_area_scaffold_activate;
|
|
|
|
g_object_class_override_property (gobject_class, PROP_ORIENTATION, "orientation");
|
|
|
|
scaffold_signals[ACTIVATE] =
|
|
g_signal_new ("activate",
|
|
G_OBJECT_CLASS_TYPE (gobject_class),
|
|
G_SIGNAL_RUN_FIRST | G_SIGNAL_ACTION,
|
|
G_STRUCT_OFFSET (CellAreaScaffoldClass, activate),
|
|
NULL, NULL,
|
|
g_cclosure_marshal_VOID__VOID,
|
|
G_TYPE_NONE, 0);
|
|
widget_class->activate_signal = scaffold_signals[ACTIVATE];
|
|
|
|
|
|
g_type_class_add_private (gobject_class, sizeof (CellAreaScaffoldPrivate));
|
|
}
|
|
|
|
/*********************************************************
|
|
* GObjectClass *
|
|
*********************************************************/
|
|
static void
|
|
cell_area_scaffold_finalize (GObject *object)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (object);
|
|
CellAreaScaffoldPrivate *priv;
|
|
|
|
priv = scaffold->priv;
|
|
|
|
g_array_free (priv->row_data, TRUE);
|
|
|
|
G_OBJECT_CLASS (cell_area_scaffold_parent_class)->finalize (object);
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_dispose (GObject *object)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (object);
|
|
CellAreaScaffoldPrivate *priv;
|
|
|
|
priv = scaffold->priv;
|
|
|
|
cell_area_scaffold_set_model (scaffold, NULL);
|
|
|
|
if (priv->iter)
|
|
{
|
|
/* Disconnect signals */
|
|
g_signal_handler_disconnect (priv->iter, priv->size_changed_id);
|
|
|
|
g_object_unref (priv->iter);
|
|
priv->iter = NULL;
|
|
priv->size_changed_id = 0;
|
|
}
|
|
|
|
if (priv->area)
|
|
{
|
|
/* Disconnect signals */
|
|
g_object_unref (priv->area);
|
|
priv->area = NULL;
|
|
}
|
|
|
|
G_OBJECT_CLASS (cell_area_scaffold_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_set_property (GObject *object,
|
|
guint prop_id,
|
|
const GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (object);
|
|
CellAreaScaffoldPrivate *priv;
|
|
|
|
priv = scaffold->priv;
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_ORIENTATION:
|
|
gtk_orientable_set_orientation (GTK_ORIENTABLE (priv->area),
|
|
g_value_get_enum (value));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_get_property (GObject *object,
|
|
guint prop_id,
|
|
GValue *value,
|
|
GParamSpec *pspec)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (object);
|
|
CellAreaScaffoldPrivate *priv;
|
|
|
|
priv = scaffold->priv;
|
|
|
|
switch (prop_id)
|
|
{
|
|
case PROP_ORIENTATION:
|
|
g_value_set_enum (value,
|
|
gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area)));
|
|
break;
|
|
default:
|
|
G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
|
|
break;
|
|
}
|
|
}
|
|
|
|
/*********************************************************
|
|
* GtkWidgetClass *
|
|
*********************************************************/
|
|
static void
|
|
cell_area_scaffold_realize (GtkWidget *widget)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkAllocation allocation;
|
|
GdkWindow *window;
|
|
GdkWindowAttr attributes;
|
|
gint attributes_mask;
|
|
|
|
gtk_widget_get_allocation (widget, &allocation);
|
|
|
|
gtk_widget_set_realized (widget, TRUE);
|
|
|
|
attributes.window_type = GDK_WINDOW_CHILD;
|
|
attributes.x = allocation.x;
|
|
attributes.y = allocation.y;
|
|
attributes.width = allocation.width;
|
|
attributes.height = allocation.height;
|
|
attributes.wclass = GDK_INPUT_ONLY;
|
|
attributes.event_mask = gtk_widget_get_events (widget);
|
|
attributes.event_mask |= (GDK_BUTTON_PRESS_MASK |
|
|
GDK_BUTTON_RELEASE_MASK |
|
|
GDK_KEY_PRESS_MASK |
|
|
GDK_KEY_RELEASE_MASK);
|
|
|
|
attributes_mask = GDK_WA_X | GDK_WA_Y;
|
|
|
|
window = gtk_widget_get_parent_window (widget);
|
|
gtk_widget_set_window (widget, window);
|
|
g_object_ref (window);
|
|
|
|
priv->event_window = gdk_window_new (window,
|
|
&attributes, attributes_mask);
|
|
gdk_window_set_user_data (priv->event_window, widget);
|
|
|
|
gtk_widget_style_attach (widget);
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_unrealize (GtkWidget *widget)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
|
|
if (priv->event_window)
|
|
{
|
|
gdk_window_set_user_data (priv->event_window, NULL);
|
|
gdk_window_destroy (priv->event_window);
|
|
priv->event_window = NULL;
|
|
}
|
|
|
|
GTK_WIDGET_CLASS (cell_area_scaffold_parent_class)->unrealize (widget);
|
|
}
|
|
|
|
static gboolean
|
|
cell_area_scaffold_draw (GtkWidget *widget,
|
|
cairo_t *cr)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkOrientation orientation;
|
|
GtkTreeIter iter;
|
|
gboolean valid;
|
|
GdkRectangle render_area;
|
|
GtkAllocation allocation;
|
|
gint i = 0;
|
|
gboolean have_focus;
|
|
GtkCellRendererState flags;
|
|
|
|
if (!priv->model)
|
|
return FALSE;
|
|
|
|
have_focus = gtk_widget_has_focus (widget);
|
|
orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
|
|
|
|
gtk_widget_get_allocation (widget, &allocation);
|
|
|
|
render_area.x = 0;
|
|
render_area.y = 0;
|
|
render_area.width = allocation.width;
|
|
render_area.height = allocation.height;
|
|
|
|
valid = gtk_tree_model_get_iter_first (priv->model, &iter);
|
|
while (valid)
|
|
{
|
|
RowData *data = &g_array_index (priv->row_data, RowData, i);
|
|
|
|
if (have_focus && i == priv->focus_row)
|
|
flags = GTK_CELL_RENDERER_FOCUSED;
|
|
else
|
|
flags = 0;
|
|
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
{
|
|
render_area.height = data->size;
|
|
}
|
|
else
|
|
{
|
|
render_area.width = data->size;
|
|
}
|
|
|
|
gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
|
|
gtk_cell_area_render (priv->area, priv->iter, widget, cr,
|
|
&render_area, &render_area, flags,
|
|
(have_focus && i == priv->focus_row));
|
|
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
{
|
|
render_area.y += data->size;
|
|
render_area.y += ROW_SPACING;
|
|
}
|
|
else
|
|
{
|
|
render_area.x += data->size;
|
|
render_area.x += ROW_SPACING;
|
|
}
|
|
|
|
i++;
|
|
valid = gtk_tree_model_iter_next (priv->model, &iter);
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
request_all_base (CellAreaScaffold *scaffold)
|
|
{
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkWidget *widget = GTK_WIDGET (scaffold);
|
|
GtkOrientation orientation;
|
|
GtkTreeIter iter;
|
|
gboolean valid;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
g_signal_handler_block (priv->iter, priv->size_changed_id);
|
|
|
|
orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
|
|
|
|
valid = gtk_tree_model_get_iter_first (priv->model, &iter);
|
|
while (valid)
|
|
{
|
|
gint min, nat;
|
|
|
|
gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
|
|
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
gtk_cell_area_get_preferred_width (priv->area, priv->iter, widget, &min, &nat);
|
|
else
|
|
gtk_cell_area_get_preferred_height (priv->area, priv->iter, widget, &min, &nat);
|
|
|
|
valid = gtk_tree_model_iter_next (priv->model, &iter);
|
|
}
|
|
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
gtk_cell_area_iter_sum_preferred_width (priv->iter);
|
|
else
|
|
gtk_cell_area_iter_sum_preferred_height (priv->iter);
|
|
|
|
g_signal_handler_unblock (priv->iter, priv->size_changed_id);
|
|
}
|
|
|
|
static void
|
|
get_row_sizes (CellAreaScaffold *scaffold,
|
|
GArray *array,
|
|
gint for_size)
|
|
{
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkWidget *widget = GTK_WIDGET (scaffold);
|
|
GtkOrientation orientation;
|
|
GtkTreeIter iter;
|
|
gboolean valid;
|
|
gint i = 0;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
|
|
|
|
valid = gtk_tree_model_get_iter_first (priv->model, &iter);
|
|
while (valid)
|
|
{
|
|
RowData *data = &g_array_index (array, RowData, i);
|
|
|
|
gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
|
|
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
gtk_cell_area_get_preferred_height_for_width (priv->area, priv->iter, widget,
|
|
for_size, &data->size, NULL);
|
|
else
|
|
gtk_cell_area_get_preferred_width_for_height (priv->area, priv->iter, widget,
|
|
for_size, &data->size, NULL);
|
|
|
|
i++;
|
|
valid = gtk_tree_model_iter_next (priv->model, &iter);
|
|
}
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_size_allocate (GtkWidget *widget,
|
|
GtkAllocation *allocation)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkOrientation orientation;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
gtk_widget_set_allocation (widget, allocation);
|
|
|
|
if (gtk_widget_get_realized (widget))
|
|
gdk_window_move_resize (priv->event_window,
|
|
allocation->x,
|
|
allocation->y,
|
|
allocation->width,
|
|
allocation->height);
|
|
|
|
orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
|
|
|
|
/* Cache the per-row sizes and allocate the iter */
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
{
|
|
get_row_sizes (scaffold, priv->row_data, allocation->width);
|
|
gtk_cell_area_iter_allocate_width (priv->iter, allocation->width);
|
|
}
|
|
else
|
|
{
|
|
get_row_sizes (scaffold, priv->row_data, allocation->height);
|
|
gtk_cell_area_iter_allocate_height (priv->iter, allocation->height);
|
|
}
|
|
}
|
|
|
|
|
|
static void
|
|
cell_area_scaffold_get_preferred_width (GtkWidget *widget,
|
|
gint *minimum_size,
|
|
gint *natural_size)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkOrientation orientation;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
|
|
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
{
|
|
request_all_base (scaffold);
|
|
|
|
gtk_cell_area_iter_get_preferred_width (priv->iter, minimum_size, natural_size);
|
|
}
|
|
else
|
|
{
|
|
gint min_size, nat_size;
|
|
|
|
GTK_WIDGET_GET_CLASS (widget)->get_preferred_height (widget, &min_size, &nat_size);
|
|
GTK_WIDGET_GET_CLASS (widget)->get_preferred_width_for_height (widget, min_size,
|
|
minimum_size, natural_size);
|
|
}
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_get_preferred_height_for_width (GtkWidget *widget,
|
|
gint for_size,
|
|
gint *minimum_size,
|
|
gint *natural_size)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkOrientation orientation;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
|
|
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
{
|
|
GArray *request_array;
|
|
gint n_rows, i, full_size = 0;
|
|
|
|
n_rows = gtk_tree_model_iter_n_children (priv->model, NULL);
|
|
|
|
/* Get an array for the contextual request */
|
|
request_array = g_array_new (FALSE, FALSE, sizeof (RowData));
|
|
g_array_set_size (request_array, n_rows);
|
|
memset (request_array->data, 0x0, n_rows * sizeof (RowData));
|
|
|
|
/* Gather each contextual size into the request array */
|
|
get_row_sizes (scaffold, request_array, for_size);
|
|
|
|
/* Sum up the size and add some row spacing */
|
|
for (i = 0; i < n_rows; i++)
|
|
{
|
|
RowData *data = &g_array_index (request_array, RowData, i);
|
|
|
|
full_size += data->size;
|
|
}
|
|
|
|
full_size += MAX (0, n_rows -1) * ROW_SPACING;
|
|
|
|
g_array_free (request_array, TRUE);
|
|
|
|
*minimum_size = full_size;
|
|
*natural_size = full_size;
|
|
}
|
|
else
|
|
{
|
|
GTK_WIDGET_GET_CLASS (widget)->get_preferred_height (widget, minimum_size, natural_size);
|
|
}
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_get_preferred_height (GtkWidget *widget,
|
|
gint *minimum_size,
|
|
gint *natural_size)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkOrientation orientation;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
|
|
|
|
if (orientation == GTK_ORIENTATION_VERTICAL)
|
|
{
|
|
request_all_base (scaffold);
|
|
|
|
gtk_cell_area_iter_get_preferred_height (priv->iter, minimum_size, natural_size);
|
|
}
|
|
else
|
|
{
|
|
gint min_size, nat_size;
|
|
|
|
GTK_WIDGET_GET_CLASS (widget)->get_preferred_width (widget, &min_size, &nat_size);
|
|
GTK_WIDGET_GET_CLASS (widget)->get_preferred_height_for_width (widget, min_size,
|
|
minimum_size, natural_size);
|
|
}
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_get_preferred_width_for_height (GtkWidget *widget,
|
|
gint for_size,
|
|
gint *minimum_size,
|
|
gint *natural_size)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkOrientation orientation;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
|
|
|
|
if (orientation == GTK_ORIENTATION_VERTICAL)
|
|
{
|
|
GArray *request_array;
|
|
gint n_rows, i, full_size = 0;
|
|
|
|
n_rows = gtk_tree_model_iter_n_children (priv->model, NULL);
|
|
|
|
/* Get an array for the contextual request */
|
|
request_array = g_array_new (FALSE, FALSE, sizeof (RowData));
|
|
g_array_set_size (request_array, n_rows);
|
|
memset (request_array->data, 0x0, n_rows * sizeof (RowData));
|
|
|
|
/* Gather each contextual size into the request array */
|
|
get_row_sizes (scaffold, request_array, for_size);
|
|
|
|
/* Sum up the size and add some row spacing */
|
|
for (i = 0; i < n_rows; i++)
|
|
{
|
|
RowData *data = &g_array_index (request_array, RowData, i);
|
|
|
|
full_size += data->size;
|
|
}
|
|
|
|
full_size += MAX (0, n_rows -1) * ROW_SPACING;
|
|
|
|
g_array_free (request_array, TRUE);
|
|
|
|
*minimum_size = full_size;
|
|
*natural_size = full_size;
|
|
}
|
|
else
|
|
{
|
|
GTK_WIDGET_GET_CLASS (widget)->get_preferred_width (widget, minimum_size, natural_size);
|
|
}
|
|
}
|
|
|
|
static gint
|
|
cell_area_scaffold_focus (GtkWidget *widget,
|
|
GtkDirectionType direction)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkTreeIter iter;
|
|
gboolean valid;
|
|
gint focus_row;
|
|
GtkOrientation orientation;
|
|
|
|
/* Grab focus on ourself if we dont already have focus */
|
|
if (!gtk_widget_has_focus (widget))
|
|
gtk_widget_grab_focus (widget);
|
|
|
|
/* Move focus from cell to cell and row to row */
|
|
orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
|
|
|
|
focus_row = priv->focus_row;
|
|
|
|
valid = gtk_tree_model_iter_nth_child (priv->model, &iter, NULL, priv->focus_row);
|
|
while (valid)
|
|
{
|
|
gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
|
|
|
|
/* If focus stays in the area we dont need to do any more */
|
|
if (gtk_cell_area_focus (priv->area, direction))
|
|
{
|
|
priv->focus_row = focus_row;
|
|
|
|
/* XXX A smarter implementation would only invalidate the rectangles where
|
|
* focus was removed from and new focus was placed */
|
|
gtk_widget_queue_draw (widget);
|
|
return TRUE;
|
|
}
|
|
else
|
|
{
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
{
|
|
if (direction == GTK_DIR_RIGHT ||
|
|
direction == GTK_DIR_LEFT)
|
|
break;
|
|
else if (direction == GTK_DIR_UP ||
|
|
direction == GTK_DIR_TAB_BACKWARD)
|
|
{
|
|
if (focus_row == 0)
|
|
break;
|
|
else
|
|
{
|
|
/* XXX A real implementation should check if the
|
|
* previous row can focus with it's attributes setup */
|
|
focus_row--;
|
|
valid = gtk_tree_model_iter_nth_child (priv->model, &iter, NULL, focus_row);
|
|
}
|
|
}
|
|
else /* direction == GTK_DIR_DOWN || GTK_DIR_TAB_FORWARD */
|
|
{
|
|
if (focus_row == priv->row_data->len - 1)
|
|
break;
|
|
else
|
|
{
|
|
/* XXX A real implementation should check if the
|
|
* previous row can focus with it's attributes setup */
|
|
focus_row++;
|
|
valid = gtk_tree_model_iter_next (priv->model, &iter);
|
|
}
|
|
}
|
|
}
|
|
else /* (orientation == GTK_ORIENTATION_HORIZONTAL) */
|
|
{
|
|
if (direction == GTK_DIR_UP ||
|
|
direction == GTK_DIR_DOWN)
|
|
break;
|
|
else if (direction == GTK_DIR_LEFT ||
|
|
direction == GTK_DIR_TAB_BACKWARD)
|
|
{
|
|
if (focus_row == 0)
|
|
break;
|
|
else
|
|
{
|
|
/* XXX A real implementation should check if the
|
|
* previous row can focus with it's attributes setup */
|
|
focus_row--;
|
|
valid = gtk_tree_model_iter_nth_child (priv->model, &iter, NULL, focus_row);
|
|
}
|
|
}
|
|
else /* direction == GTK_DIR_RIGHT || GTK_DIR_TAB_FORWARD */
|
|
{
|
|
if (focus_row == priv->row_data->len - 1)
|
|
break;
|
|
else
|
|
{
|
|
/* XXX A real implementation should check if the
|
|
* previous row can focus with it's attributes setup */
|
|
focus_row++;
|
|
valid = gtk_tree_model_iter_next (priv->model, &iter);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
/* XXX A smarter implementation would only invalidate the rectangles where
|
|
* focus was removed from and new focus was placed */
|
|
gtk_widget_queue_draw (widget);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
/*********************************************************
|
|
* CellAreaScaffoldClass *
|
|
*********************************************************/
|
|
static void
|
|
cell_area_scaffold_activate (CellAreaScaffold *scaffold)
|
|
{
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkWidget *widget = GTK_WIDGET (scaffold);
|
|
GtkAllocation allocation;
|
|
GtkOrientation orientation;
|
|
GdkRectangle cell_area;
|
|
GtkTreeIter iter;
|
|
gboolean valid;
|
|
gint i = 0;
|
|
|
|
orientation = gtk_orientable_get_orientation (GTK_ORIENTABLE (priv->area));
|
|
gtk_widget_get_allocation (widget, &allocation);
|
|
|
|
cell_area.x = 0;
|
|
cell_area.y = 0;
|
|
cell_area.width = allocation.width;
|
|
cell_area.height = allocation.height;
|
|
|
|
valid = gtk_tree_model_get_iter_first (priv->model, &iter);
|
|
while (valid)
|
|
{
|
|
RowData *data = &g_array_index (priv->row_data, RowData, i);
|
|
|
|
if (i == priv->focus_row)
|
|
{
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
cell_area.height = data->size;
|
|
else
|
|
cell_area.width = data->size;
|
|
|
|
gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
|
|
gtk_cell_area_activate (priv->area, priv->iter, widget, &cell_area, GTK_CELL_RENDERER_FOCUSED);
|
|
|
|
break;
|
|
}
|
|
|
|
if (orientation == GTK_ORIENTATION_HORIZONTAL)
|
|
cell_area.y += data->size + ROW_SPACING;
|
|
else
|
|
cell_area.x += data->size + ROW_SPACING;
|
|
|
|
i++;
|
|
valid = gtk_tree_model_iter_next (priv->model, &iter);
|
|
}
|
|
}
|
|
|
|
/*********************************************************
|
|
* CellArea/GtkTreeModel callbacks *
|
|
*********************************************************/
|
|
static void
|
|
size_changed_cb (GtkCellAreaIter *iter,
|
|
GParamSpec *pspec,
|
|
CellAreaScaffold *scaffold)
|
|
{
|
|
if (!strcmp (pspec->name, "minimum-width") ||
|
|
!strcmp (pspec->name, "natural-width") ||
|
|
!strcmp (pspec->name, "minimum-height") ||
|
|
!strcmp (pspec->name, "natural-height"))
|
|
gtk_widget_queue_resize (GTK_WIDGET (scaffold));
|
|
}
|
|
|
|
static void
|
|
rebuild_and_flush_internals (CellAreaScaffold *scaffold)
|
|
{
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
gint n_rows;
|
|
|
|
if (priv->model)
|
|
{
|
|
n_rows = gtk_tree_model_iter_n_children (priv->model, NULL);
|
|
|
|
/* Clear/reset the array */
|
|
g_array_set_size (priv->row_data, n_rows);
|
|
memset (priv->row_data->data, 0x0, n_rows * sizeof (RowData));
|
|
}
|
|
else
|
|
g_array_set_size (priv->row_data, 0);
|
|
|
|
/* Data changed, lets flush the iter and consequently queue resize and
|
|
* start everything over again (note this is definitly far from optimized) */
|
|
gtk_cell_area_iter_flush (priv->iter);
|
|
}
|
|
|
|
static void
|
|
row_changed_cb (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
GtkTreeIter *iter,
|
|
CellAreaScaffold *scaffold)
|
|
{
|
|
rebuild_and_flush_internals (scaffold);
|
|
}
|
|
|
|
static void
|
|
row_inserted_cb (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
GtkTreeIter *iter,
|
|
CellAreaScaffold *scaffold)
|
|
{
|
|
rebuild_and_flush_internals (scaffold);
|
|
}
|
|
|
|
static void
|
|
row_deleted_cb (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
CellAreaScaffold *scaffold)
|
|
{
|
|
rebuild_and_flush_internals (scaffold);
|
|
}
|
|
|
|
static void
|
|
rows_reordered_cb (GtkTreeModel *model,
|
|
GtkTreePath *parent,
|
|
GtkTreeIter *iter,
|
|
gint *new_order,
|
|
CellAreaScaffold *scaffold)
|
|
{
|
|
rebuild_and_flush_internals (scaffold);
|
|
}
|
|
|
|
/*********************************************************
|
|
* API *
|
|
*********************************************************/
|
|
GtkWidget *
|
|
cell_area_scaffold_new (void)
|
|
{
|
|
return (GtkWidget *)g_object_new (TYPE_CELL_AREA_SCAFFOLD, NULL);
|
|
}
|
|
|
|
GtkCellArea *
|
|
cell_area_scaffold_get_area (CellAreaScaffold *scaffold)
|
|
{
|
|
CellAreaScaffoldPrivate *priv;
|
|
|
|
g_return_val_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold), NULL);
|
|
|
|
priv = scaffold->priv;
|
|
|
|
return priv->area;
|
|
}
|
|
|
|
void
|
|
cell_area_scaffold_set_model (CellAreaScaffold *scaffold,
|
|
GtkTreeModel *model)
|
|
{
|
|
CellAreaScaffoldPrivate *priv;
|
|
|
|
g_return_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold));
|
|
|
|
priv = scaffold->priv;
|
|
|
|
if (priv->model != model)
|
|
{
|
|
if (priv->model)
|
|
{
|
|
g_signal_handler_disconnect (priv->model, priv->row_changed_id);
|
|
g_signal_handler_disconnect (priv->model, priv->row_inserted_id);
|
|
g_signal_handler_disconnect (priv->model, priv->row_deleted_id);
|
|
g_signal_handler_disconnect (priv->model, priv->rows_reordered_id);
|
|
|
|
g_object_unref (priv->model);
|
|
}
|
|
|
|
priv->model = model;
|
|
|
|
if (priv->model)
|
|
{
|
|
g_object_ref (priv->model);
|
|
|
|
priv->row_changed_id =
|
|
g_signal_connect (priv->model, "row-changed",
|
|
G_CALLBACK (row_changed_cb), scaffold);
|
|
|
|
priv->row_inserted_id =
|
|
g_signal_connect (priv->model, "row-inserted",
|
|
G_CALLBACK (row_inserted_cb), scaffold);
|
|
|
|
priv->row_deleted_id =
|
|
g_signal_connect (priv->model, "row-deleted",
|
|
G_CALLBACK (row_deleted_cb), scaffold);
|
|
|
|
priv->rows_reordered_id =
|
|
g_signal_connect (priv->model, "rows-reordered",
|
|
G_CALLBACK (rows_reordered_cb), scaffold);
|
|
}
|
|
|
|
rebuild_and_flush_internals (scaffold);
|
|
}
|
|
}
|
|
|
|
GtkTreeModel *
|
|
cell_area_scaffold_get_model (CellAreaScaffold *scaffold)
|
|
{
|
|
CellAreaScaffoldPrivate *priv;
|
|
|
|
g_return_val_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold), NULL);
|
|
|
|
priv = scaffold->priv;
|
|
|
|
return priv->model;
|
|
}
|