mirror of
https://gitlab.gnome.org/GNOME/gtk.git
synced 2024-11-12 03:40:10 +00:00
2f4e451075
This argument allows the caller to specify that only an editable cell should start editing but an activatable cell should not toggle it's state, this is important for public apis like gtk_tree_view_set_cursor_on_cell() which are only intended to programatically bring attention to the editing of a specific row or cell but not actually change any data. GtkTreeView & CellAreaScaffold updated for the last minute api change.
1141 lines
34 KiB
C
1141 lines
34 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);
|
|
|
|
/* 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 void cell_area_scaffold_map (GtkWidget *widget);
|
|
static void cell_area_scaffold_unmap (GtkWidget *widget);
|
|
static gint cell_area_scaffold_focus (GtkWidget *widget,
|
|
GtkDirectionType direction);
|
|
static gboolean cell_area_scaffold_button_press (GtkWidget *widget,
|
|
GdkEventButton *event);
|
|
|
|
/* GtkContainerClass */
|
|
static void cell_area_scaffold_forall (GtkContainer *container,
|
|
gboolean include_internals,
|
|
GtkCallback callback,
|
|
gpointer callback_data);
|
|
static void cell_area_scaffold_remove (GtkContainer *container,
|
|
GtkWidget *child);
|
|
static void cell_area_scaffold_put_edit_widget (CellAreaScaffold *scaffold,
|
|
GtkWidget *edit_widget,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height);
|
|
|
|
/* CellAreaScaffoldClass */
|
|
static void cell_area_scaffold_activate (CellAreaScaffold *scaffold);
|
|
|
|
/* CellArea/GtkTreeModel callbacks */
|
|
static void size_changed_cb (GtkCellAreaContext *context,
|
|
GParamSpec *pspec,
|
|
CellAreaScaffold *scaffold);
|
|
static void focus_changed_cb (GtkCellArea *area,
|
|
GtkCellRenderer *renderer,
|
|
const gchar *path,
|
|
CellAreaScaffold *scaffold);
|
|
static void add_editable_cb (GtkCellArea *area,
|
|
GtkCellRenderer *renderer,
|
|
GtkCellEditable *edit_widget,
|
|
GdkRectangle *cell_area,
|
|
const gchar *path,
|
|
CellAreaScaffold *scaffold);
|
|
static void remove_editable_cb (GtkCellArea *area,
|
|
GtkCellRenderer *renderer,
|
|
GtkCellEditable *edit_widget,
|
|
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 height of rows in the scaffold's */
|
|
} 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 context */
|
|
GtkCellArea *area;
|
|
GtkCellAreaContext *context;
|
|
|
|
/* Cache some info about rows (hieghts etc) */
|
|
GArray *row_data;
|
|
|
|
/* Focus handling */
|
|
gint focus_row;
|
|
gulong focus_changed_id;
|
|
|
|
/* Check when the underlying area changes the size and
|
|
* we need to queue a redraw */
|
|
gulong size_changed_id;
|
|
|
|
/* Currently edited widget */
|
|
GtkWidget *edit_widget;
|
|
GdkRectangle edit_rect;
|
|
gulong add_editable_id;
|
|
gulong remove_editable_id;
|
|
|
|
|
|
gint row_spacing;
|
|
gint indent;
|
|
};
|
|
|
|
enum {
|
|
ACTIVATE,
|
|
N_SIGNALS
|
|
};
|
|
|
|
static guint scaffold_signals[N_SIGNALS] = { 0 };
|
|
|
|
#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_CONTAINER,
|
|
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->context = gtk_cell_area_create_context (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->context, "notify",
|
|
G_CALLBACK (size_changed_cb), scaffold);
|
|
|
|
priv->focus_changed_id =
|
|
g_signal_connect (priv->area, "focus-changed",
|
|
G_CALLBACK (focus_changed_cb), scaffold);
|
|
|
|
priv->add_editable_id =
|
|
g_signal_connect (priv->area, "add-editable",
|
|
G_CALLBACK (add_editable_cb), scaffold);
|
|
|
|
priv->remove_editable_id =
|
|
g_signal_connect (priv->area, "remove-editable",
|
|
G_CALLBACK (remove_editable_cb), scaffold);
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_class_init (CellAreaScaffoldClass *class)
|
|
{
|
|
GObjectClass *gobject_class;
|
|
GtkWidgetClass *widget_class;
|
|
GtkContainerClass *container_class;
|
|
|
|
gobject_class = G_OBJECT_CLASS (class);
|
|
gobject_class->dispose = cell_area_scaffold_dispose;
|
|
gobject_class->finalize = cell_area_scaffold_finalize;
|
|
|
|
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->map = cell_area_scaffold_map;
|
|
widget_class->unmap = cell_area_scaffold_unmap;
|
|
widget_class->focus = cell_area_scaffold_focus;
|
|
widget_class->button_press_event = cell_area_scaffold_button_press;
|
|
|
|
container_class = GTK_CONTAINER_CLASS (class);
|
|
container_class->forall = cell_area_scaffold_forall;
|
|
container_class->remove = cell_area_scaffold_remove;
|
|
|
|
class->activate = cell_area_scaffold_activate;
|
|
|
|
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->context)
|
|
{
|
|
/* Disconnect signals */
|
|
g_signal_handler_disconnect (priv->context, priv->size_changed_id);
|
|
|
|
g_object_unref (priv->context);
|
|
priv->context = NULL;
|
|
priv->size_changed_id = 0;
|
|
}
|
|
|
|
if (priv->area)
|
|
{
|
|
/* Disconnect signals */
|
|
g_signal_handler_disconnect (priv->area, priv->focus_changed_id);
|
|
g_signal_handler_disconnect (priv->area, priv->add_editable_id);
|
|
g_signal_handler_disconnect (priv->area, priv->remove_editable_id);
|
|
|
|
g_object_unref (priv->area);
|
|
priv->area = NULL;
|
|
priv->focus_changed_id = 0;
|
|
}
|
|
|
|
G_OBJECT_CLASS (cell_area_scaffold_parent_class)->dispose (object);
|
|
}
|
|
|
|
/*********************************************************
|
|
* 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;
|
|
GtkTreeIter iter;
|
|
gboolean valid;
|
|
GdkRectangle background_area;
|
|
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);
|
|
|
|
gtk_widget_get_allocation (widget, &allocation);
|
|
|
|
render_area.x = 0;
|
|
render_area.y = 0;
|
|
render_area.width = allocation.width;
|
|
render_area.height = allocation.height;
|
|
|
|
background_area = render_area;
|
|
|
|
render_area.x = priv->indent;
|
|
render_area.width -= priv->indent;
|
|
|
|
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;
|
|
|
|
render_area.height = data->size;
|
|
|
|
background_area.height = render_area.height;
|
|
background_area.y = render_area.y;
|
|
|
|
if (i == 0)
|
|
{
|
|
background_area.height += priv->row_spacing / 2;
|
|
background_area.height += priv->row_spacing % 2;
|
|
}
|
|
else if (i == priv->row_data->len - 1)
|
|
{
|
|
background_area.y -= priv->row_spacing / 2;
|
|
background_area.height += priv->row_spacing / 2;
|
|
}
|
|
else
|
|
{
|
|
background_area.y -= priv->row_spacing / 2;
|
|
background_area.height += priv->row_spacing;
|
|
}
|
|
|
|
gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
|
|
gtk_cell_area_render (priv->area, priv->context, widget, cr,
|
|
&background_area, &render_area, flags,
|
|
(have_focus && i == priv->focus_row));
|
|
|
|
render_area.y += data->size;
|
|
render_area.y += priv->row_spacing;
|
|
|
|
i++;
|
|
valid = gtk_tree_model_iter_next (priv->model, &iter);
|
|
}
|
|
|
|
/* Draw the edit widget after drawing everything else */
|
|
GTK_WIDGET_CLASS (cell_area_scaffold_parent_class)->draw (widget, cr);
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
request_all_base (CellAreaScaffold *scaffold)
|
|
{
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkWidget *widget = GTK_WIDGET (scaffold);
|
|
GtkTreeIter iter;
|
|
gboolean valid;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
g_signal_handler_block (priv->context, priv->size_changed_id);
|
|
|
|
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);
|
|
gtk_cell_area_get_preferred_width (priv->area, priv->context, widget, &min, &nat);
|
|
|
|
valid = gtk_tree_model_iter_next (priv->model, &iter);
|
|
}
|
|
|
|
g_signal_handler_unblock (priv->context, 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);
|
|
GtkTreeIter iter;
|
|
gboolean valid;
|
|
gint i = 0;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
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);
|
|
gtk_cell_area_get_preferred_height_for_width (priv->area, priv->context, 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;
|
|
|
|
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);
|
|
|
|
/* Allocate the child GtkCellEditable widget if one is currently editing a row */
|
|
if (priv->edit_widget)
|
|
gtk_widget_size_allocate (priv->edit_widget, &priv->edit_rect);
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
/* Cache the per-row sizes and allocate the context */
|
|
gtk_cell_area_context_allocate (priv->context, allocation->width - priv->indent, -1);
|
|
get_row_sizes (scaffold, priv->row_data, allocation->width - priv->indent);
|
|
}
|
|
|
|
|
|
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;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
request_all_base (scaffold);
|
|
gtk_cell_area_context_get_preferred_width (priv->context, minimum_size, natural_size);
|
|
|
|
*minimum_size += priv->indent;
|
|
*natural_size += priv->indent;
|
|
}
|
|
|
|
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;
|
|
GArray *request_array;
|
|
gint n_rows, i, full_size = 0;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
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 - priv->indent);
|
|
|
|
/* 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) * priv->row_spacing;
|
|
|
|
g_array_free (request_array, TRUE);
|
|
|
|
*minimum_size = full_size;
|
|
*natural_size = full_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;
|
|
gint min_size, nat_size;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
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;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
GTK_WIDGET_GET_CLASS (widget)->get_preferred_width (widget, minimum_size, natural_size);
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_map (GtkWidget *widget)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
|
|
GTK_WIDGET_CLASS (cell_area_scaffold_parent_class)->map (widget);
|
|
|
|
if (priv->event_window)
|
|
gdk_window_show (priv->event_window);
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_unmap (GtkWidget *widget)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
|
|
GTK_WIDGET_CLASS (cell_area_scaffold_parent_class)->unmap (widget);
|
|
|
|
if (priv->event_window)
|
|
gdk_window_hide (priv->event_window);
|
|
}
|
|
|
|
|
|
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;
|
|
gboolean changed = FALSE;
|
|
|
|
/* 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 */
|
|
focus_row = priv->focus_row;
|
|
|
|
g_signal_handler_block (priv->area, priv->focus_changed_id);
|
|
|
|
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);
|
|
changed = TRUE;
|
|
break;
|
|
}
|
|
else
|
|
{
|
|
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);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
g_signal_handler_unblock (priv->area, priv->focus_changed_id);
|
|
|
|
/* 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 changed;
|
|
}
|
|
|
|
static gboolean
|
|
cell_area_scaffold_button_press (GtkWidget *widget,
|
|
GdkEventButton *event)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (widget);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkTreeIter iter;
|
|
gboolean valid;
|
|
gint i = 0;
|
|
GdkRectangle event_area;
|
|
GtkAllocation allocation;
|
|
gboolean handled = FALSE;
|
|
|
|
gtk_widget_get_allocation (widget, &allocation);
|
|
|
|
event_area.x = 0;
|
|
event_area.y = 0;
|
|
event_area.width = allocation.width;
|
|
event_area.height = allocation.height;
|
|
|
|
event_area.x = priv->indent;
|
|
event_area.width -= priv->indent;
|
|
|
|
valid = gtk_tree_model_get_iter_first (priv->model, &iter);
|
|
while (valid)
|
|
{
|
|
RowData *data = &g_array_index (priv->row_data, RowData, i);
|
|
|
|
event_area.height = data->size;
|
|
|
|
if (event->y >= event_area.y &&
|
|
event->y <= event_area.y + event_area.height)
|
|
{
|
|
/* XXX A real implementation would assemble GtkCellRendererState flags here */
|
|
gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
|
|
handled = gtk_cell_area_event (priv->area, priv->context, GTK_WIDGET (scaffold),
|
|
(GdkEvent *)event, &event_area, 0);
|
|
break;
|
|
}
|
|
|
|
event_area.y += data->size;
|
|
event_area.y += priv->row_spacing;
|
|
|
|
i++;
|
|
valid = gtk_tree_model_iter_next (priv->model, &iter);
|
|
}
|
|
|
|
return handled;
|
|
}
|
|
|
|
|
|
/*********************************************************
|
|
* GtkContainerClass *
|
|
*********************************************************/
|
|
static void
|
|
cell_area_scaffold_put_edit_widget (CellAreaScaffold *scaffold,
|
|
GtkWidget *edit_widget,
|
|
gint x,
|
|
gint y,
|
|
gint width,
|
|
gint height)
|
|
{
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
|
|
priv->edit_rect.x = x;
|
|
priv->edit_rect.y = y;
|
|
priv->edit_rect.width = width;
|
|
priv->edit_rect.height = height;
|
|
priv->edit_widget = edit_widget;
|
|
|
|
gtk_widget_set_parent (edit_widget, GTK_WIDGET (scaffold));
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_forall (GtkContainer *container,
|
|
gboolean include_internals,
|
|
GtkCallback callback,
|
|
gpointer callback_data)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (container);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
|
|
if (priv->edit_widget)
|
|
(* callback) (priv->edit_widget, callback_data);
|
|
}
|
|
|
|
static void
|
|
cell_area_scaffold_remove (GtkContainer *container,
|
|
GtkWidget *child)
|
|
{
|
|
CellAreaScaffold *scaffold = CELL_AREA_SCAFFOLD (container);
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
|
|
g_return_if_fail (child == priv->edit_widget);
|
|
|
|
gtk_widget_unparent (priv->edit_widget);
|
|
priv->edit_widget = NULL;
|
|
}
|
|
|
|
/*********************************************************
|
|
* CellAreaScaffoldClass *
|
|
*********************************************************/
|
|
static void
|
|
cell_area_scaffold_activate (CellAreaScaffold *scaffold)
|
|
{
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkWidget *widget = GTK_WIDGET (scaffold);
|
|
GtkAllocation allocation;
|
|
GdkRectangle cell_area;
|
|
GtkTreeIter iter;
|
|
gboolean valid;
|
|
gint i = 0;
|
|
|
|
gtk_widget_get_allocation (widget, &allocation);
|
|
|
|
cell_area.x = 0;
|
|
cell_area.y = 0;
|
|
cell_area.width = allocation.width;
|
|
cell_area.height = allocation.height;
|
|
|
|
cell_area.x = priv->indent;
|
|
cell_area.width -= priv->indent;
|
|
|
|
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)
|
|
{
|
|
cell_area.height = data->size;
|
|
|
|
gtk_cell_area_apply_attributes (priv->area, priv->model, &iter, FALSE, FALSE);
|
|
gtk_cell_area_activate (priv->area, priv->context, widget, &cell_area,
|
|
GTK_CELL_RENDERER_FOCUSED, FALSE);
|
|
|
|
break;
|
|
}
|
|
|
|
cell_area.y += data->size + priv->row_spacing;
|
|
|
|
i++;
|
|
valid = gtk_tree_model_iter_next (priv->model, &iter);
|
|
}
|
|
}
|
|
|
|
/*********************************************************
|
|
* CellArea/GtkTreeModel callbacks *
|
|
*********************************************************/
|
|
static void
|
|
size_changed_cb (GtkCellAreaContext *context,
|
|
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
|
|
focus_changed_cb (GtkCellArea *area,
|
|
GtkCellRenderer *renderer,
|
|
const gchar *path,
|
|
CellAreaScaffold *scaffold)
|
|
{
|
|
CellAreaScaffoldPrivate *priv = scaffold->priv;
|
|
GtkWidget *widget = GTK_WIDGET (scaffold);
|
|
GtkTreePath *treepath;
|
|
gint *indices;
|
|
|
|
if (!priv->model)
|
|
return;
|
|
|
|
/* We can be signaled that a renderer lost focus, here
|
|
* we dont care */
|
|
if (!renderer)
|
|
return;
|
|
|
|
treepath = gtk_tree_path_new_from_string (path);
|
|
indices = gtk_tree_path_get_indices (treepath);
|
|
|
|
priv->focus_row = indices[0];
|
|
|
|
gtk_tree_path_free (treepath);
|
|
|
|
/* Make sure we have focus now */
|
|
if (!gtk_widget_has_focus (widget))
|
|
gtk_widget_grab_focus (widget);
|
|
|
|
gtk_widget_queue_draw (widget);
|
|
}
|
|
|
|
static void
|
|
add_editable_cb (GtkCellArea *area,
|
|
GtkCellRenderer *renderer,
|
|
GtkCellEditable *edit_widget,
|
|
GdkRectangle *cell_area,
|
|
const gchar *path,
|
|
CellAreaScaffold *scaffold)
|
|
{
|
|
GtkAllocation allocation;
|
|
|
|
gtk_widget_get_allocation (GTK_WIDGET (scaffold), &allocation);
|
|
|
|
cell_area_scaffold_put_edit_widget (scaffold, GTK_WIDGET (edit_widget),
|
|
allocation.x + cell_area->x,
|
|
allocation.y + cell_area->y,
|
|
cell_area->width, cell_area->height);
|
|
}
|
|
|
|
static void
|
|
remove_editable_cb (GtkCellArea *area,
|
|
GtkCellRenderer *renderer,
|
|
GtkCellEditable *edit_widget,
|
|
CellAreaScaffold *scaffold)
|
|
{
|
|
gtk_container_remove (GTK_CONTAINER (scaffold), GTK_WIDGET (edit_widget));
|
|
|
|
gtk_widget_grab_focus (GTK_WIDGET (scaffold));
|
|
}
|
|
|
|
static void
|
|
rebuild_and_reset_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 reset the context and consequently queue resize and
|
|
* start everything over again (note this is definitly far from optimized) */
|
|
gtk_cell_area_context_reset (priv->context);
|
|
}
|
|
|
|
static void
|
|
row_changed_cb (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
GtkTreeIter *iter,
|
|
CellAreaScaffold *scaffold)
|
|
{
|
|
rebuild_and_reset_internals (scaffold);
|
|
}
|
|
|
|
static void
|
|
row_inserted_cb (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
GtkTreeIter *iter,
|
|
CellAreaScaffold *scaffold)
|
|
{
|
|
rebuild_and_reset_internals (scaffold);
|
|
}
|
|
|
|
static void
|
|
row_deleted_cb (GtkTreeModel *model,
|
|
GtkTreePath *path,
|
|
CellAreaScaffold *scaffold)
|
|
{
|
|
rebuild_and_reset_internals (scaffold);
|
|
}
|
|
|
|
static void
|
|
rows_reordered_cb (GtkTreeModel *model,
|
|
GtkTreePath *parent,
|
|
GtkTreeIter *iter,
|
|
gint *new_order,
|
|
CellAreaScaffold *scaffold)
|
|
{
|
|
rebuild_and_reset_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_reset_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;
|
|
}
|
|
|
|
|
|
void
|
|
cell_area_scaffold_set_row_spacing (CellAreaScaffold *scaffold,
|
|
gint spacing)
|
|
{
|
|
CellAreaScaffoldPrivate *priv;
|
|
|
|
g_return_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold));
|
|
|
|
priv = scaffold->priv;
|
|
|
|
if (priv->row_spacing != spacing)
|
|
{
|
|
priv->row_spacing = spacing;
|
|
gtk_widget_queue_resize (GTK_WIDGET (scaffold));
|
|
}
|
|
}
|
|
|
|
gint
|
|
cell_area_scaffold_get_row_spacing (CellAreaScaffold *scaffold)
|
|
{
|
|
CellAreaScaffoldPrivate *priv;
|
|
|
|
g_return_val_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold), 0);
|
|
|
|
priv = scaffold->priv;
|
|
|
|
return priv->row_spacing;
|
|
}
|
|
|
|
void
|
|
cell_area_scaffold_set_indentation (CellAreaScaffold *scaffold,
|
|
gint indent)
|
|
{
|
|
CellAreaScaffoldPrivate *priv;
|
|
|
|
g_return_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold));
|
|
|
|
priv = scaffold->priv;
|
|
|
|
if (priv->indent != indent)
|
|
{
|
|
priv->indent = indent;
|
|
gtk_widget_queue_resize (GTK_WIDGET (scaffold));
|
|
}
|
|
}
|
|
|
|
gint
|
|
cell_area_scaffold_get_indentation (CellAreaScaffold *scaffold)
|
|
{
|
|
CellAreaScaffoldPrivate *priv;
|
|
|
|
g_return_val_if_fail (IS_CELL_AREA_SCAFFOLD (scaffold), 0);
|
|
|
|
priv = scaffold->priv;
|
|
|
|
return priv->indent;
|
|
}
|
|
|
|
|