+++ /dev/null
-/* GTK - The GIMP Toolkit
- * Copyright (C) 1995-1997 Peter Mattis, Spencer Kimball and Josh MacDonald
- *
- * This library 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 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
- * Lesser General Public License for more details.
- *
- * You should have received a copy of the GNU Lesser 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.
- */
-
-/*
- * Modified by the GTK+ Team and others 1997-2000. See the AUTHORS
- * file for a list of people on the GTK+ Team. See the ChangeLog
- * files for a list of changes. These files are distributed with
- * GTK+ at ftp://ftp.gtk.org/pub/gtk/.
- */
-
-#include "config.h"
-#include <string.h> /* memset */
-
-#undef GTK_DISABLE_DEPRECATED
-#define __GTK_LIST_C__
-
-#include "gtklist.h"
-#include "gtklistitem.h"
-#include "gtkmain.h"
-#include "gtksignal.h"
-#include "gtklabel.h"
-#include "gtkmarshalers.h"
-#include "gtkintl.h"
-
-#include "gtkalias.h"
-
-enum {
- SELECTION_CHANGED,
- SELECT_CHILD,
- UNSELECT_CHILD,
- LAST_SIGNAL
-};
-
-enum {
- ARG_0,
- ARG_SELECTION_MODE
-};
-
-#define SCROLL_TIME 100
-
-/*** GtkList Methods ***/
-static void gtk_list_class_init (GtkListClass *klass);
-static void gtk_list_init (GtkList *list);
-static void gtk_list_set_arg (GtkObject *object,
- GtkArg *arg,
- guint arg_id);
-static void gtk_list_get_arg (GtkObject *object,
- GtkArg *arg,
- guint arg_id);
-/*** GtkObject Methods ***/
-static void gtk_list_dispose (GObject *object);
-
-/*** GtkWidget Methods ***/
-static void gtk_list_size_request (GtkWidget *widget,
- GtkRequisition *requisition);
-static void gtk_list_size_allocate (GtkWidget *widget,
- GtkAllocation *allocation);
-static void gtk_list_realize (GtkWidget *widget);
-static void gtk_list_unmap (GtkWidget *widget);
-static void gtk_list_style_set (GtkWidget *widget,
- GtkStyle *previous_style);
-static gint gtk_list_motion_notify (GtkWidget *widget,
- GdkEventMotion *event);
-static gint gtk_list_button_press (GtkWidget *widget,
- GdkEventButton *event);
-static gint gtk_list_button_release (GtkWidget *widget,
- GdkEventButton *event);
-
-static gboolean gtk_list_focus (GtkWidget *widget,
- GtkDirectionType direction);
-
-/*** GtkContainer Methods ***/
-static void gtk_list_add (GtkContainer *container,
- GtkWidget *widget);
-static void gtk_list_remove (GtkContainer *container,
- GtkWidget *widget);
-static void gtk_list_forall (GtkContainer *container,
- gboolean include_internals,
- GtkCallback callback,
- gpointer callback_data);
-static GtkType gtk_list_child_type (GtkContainer *container);
-static void gtk_list_set_focus_child (GtkContainer *container,
- GtkWidget *widget);
-
-/*** GtkList Private Functions ***/
-static void gtk_list_move_focus_child (GtkList *list,
- GtkScrollType scroll_type,
- gfloat position);
-static gint gtk_list_horizontal_timeout (GtkWidget *list);
-static gint gtk_list_vertical_timeout (GtkWidget *list);
-static void gtk_list_remove_items_internal (GtkList *list,
- GList *items,
- gboolean no_unref);
-
-/*** GtkList Selection Methods ***/
-static void gtk_real_list_select_child (GtkList *list,
- GtkWidget *child);
-static void gtk_real_list_unselect_child (GtkList *list,
- GtkWidget *child);
-
-/*** GtkList Selection Functions ***/
-static void gtk_list_set_anchor (GtkList *list,
- gboolean add_mode,
- gint anchor,
- GtkWidget *undo_focus_child);
-static void gtk_list_fake_unselect_all (GtkList *list,
- GtkWidget *item);
-static void gtk_list_fake_toggle_row (GtkList *list,
- GtkWidget *item);
-static void gtk_list_update_extended_selection (GtkList *list,
- gint row);
-static void gtk_list_reset_extended_selection (GtkList *list);
-
-/*** GtkListItem Signal Functions ***/
-static void gtk_list_signal_drag_begin (GtkWidget *widget,
- GdkDragContext *context,
- GtkList *list);
-static void gtk_list_signal_toggle_focus_row (GtkListItem *list_item,
- GtkList *list);
-static void gtk_list_signal_select_all (GtkListItem *list_item,
- GtkList *list);
-static void gtk_list_signal_unselect_all (GtkListItem *list_item,
- GtkList *list);
-static void gtk_list_signal_undo_selection (GtkListItem *list_item,
- GtkList *list);
-static void gtk_list_signal_start_selection (GtkListItem *list_item,
- GtkList *list);
-static void gtk_list_signal_end_selection (GtkListItem *list_item,
- GtkList *list);
-static void gtk_list_signal_extend_selection (GtkListItem *list_item,
- GtkScrollType scroll_type,
- gfloat position,
- gboolean auto_start_selection,
- GtkList *list);
-static void gtk_list_signal_scroll_horizontal (GtkListItem *list_item,
- GtkScrollType scroll_type,
- gfloat position,
- GtkList *list);
-static void gtk_list_signal_scroll_vertical (GtkListItem *list_item,
- GtkScrollType scroll_type,
- gfloat position,
- GtkList *list);
-static void gtk_list_signal_toggle_add_mode (GtkListItem *list_item,
- GtkList *list);
-static void gtk_list_signal_item_select (GtkListItem *list_item,
- GtkList *list);
-static void gtk_list_signal_item_deselect (GtkListItem *list_item,
- GtkList *list);
-static void gtk_list_signal_item_toggle (GtkListItem *list_item,
- GtkList *list);
-
-
-static void gtk_list_drag_begin (GtkWidget *widget,
- GdkDragContext *context);
-
-
-static GtkContainerClass *parent_class = NULL;
-static guint list_signals[LAST_SIGNAL] = { 0 };
-
-static const gchar vadjustment_key[] = "gtk-vadjustment";
-static guint vadjustment_key_id = 0;
-static const gchar hadjustment_key[] = "gtk-hadjustment";
-static guint hadjustment_key_id = 0;
-
-GtkType
-gtk_list_get_type (void)
-{
- static GtkType list_type = 0;
-
- if (!list_type)
- {
- static const GtkTypeInfo list_info =
- {
- "GtkList",
- sizeof (GtkList),
- sizeof (GtkListClass),
- (GtkClassInitFunc) gtk_list_class_init,
- (GtkObjectInitFunc) gtk_list_init,
- /* reserved_1 */ NULL,
- /* reserved_2 */ NULL,
- (GtkClassInitFunc) NULL,
- };
-
- I_("GtkList");
- list_type = gtk_type_unique (GTK_TYPE_CONTAINER, &list_info);
- }
-
- return list_type;
-}
-
-static void
-gtk_list_class_init (GtkListClass *class)
-{
- GObjectClass *gobject_class = G_OBJECT_CLASS (class);
- GtkObjectClass *object_class;
- GtkWidgetClass *widget_class;
- GtkContainerClass *container_class;
-
- object_class = (GtkObjectClass*) class;
- widget_class = (GtkWidgetClass*) class;
- container_class = (GtkContainerClass*) class;
-
- parent_class = gtk_type_class (GTK_TYPE_CONTAINER);
-
- vadjustment_key_id = g_quark_from_static_string (vadjustment_key);
- hadjustment_key_id = g_quark_from_static_string (hadjustment_key);
-
- gobject_class->dispose = gtk_list_dispose;
-
-
- object_class->set_arg = gtk_list_set_arg;
- object_class->get_arg = gtk_list_get_arg;
-
- widget_class->unmap = gtk_list_unmap;
- widget_class->style_set = gtk_list_style_set;
- widget_class->realize = gtk_list_realize;
- widget_class->button_press_event = gtk_list_button_press;
- widget_class->button_release_event = gtk_list_button_release;
- widget_class->motion_notify_event = gtk_list_motion_notify;
- widget_class->size_request = gtk_list_size_request;
- widget_class->size_allocate = gtk_list_size_allocate;
- widget_class->drag_begin = gtk_list_drag_begin;
- widget_class->focus = gtk_list_focus;
-
- container_class->add = gtk_list_add;
- container_class->remove = gtk_list_remove;
- container_class->forall = gtk_list_forall;
- container_class->child_type = gtk_list_child_type;
- container_class->set_focus_child = gtk_list_set_focus_child;
-
- class->selection_changed = NULL;
- class->select_child = gtk_real_list_select_child;
- class->unselect_child = gtk_real_list_unselect_child;
-
- list_signals[SELECTION_CHANGED] =
- gtk_signal_new (I_("selection-changed"),
- GTK_RUN_FIRST,
- GTK_CLASS_TYPE (object_class),
- GTK_SIGNAL_OFFSET (GtkListClass, selection_changed),
- _gtk_marshal_VOID__VOID,
- GTK_TYPE_NONE, 0);
- list_signals[SELECT_CHILD] =
- gtk_signal_new (I_("select-child"),
- GTK_RUN_FIRST,
- GTK_CLASS_TYPE (object_class),
- GTK_SIGNAL_OFFSET (GtkListClass, select_child),
- _gtk_marshal_VOID__OBJECT,
- GTK_TYPE_NONE, 1,
- GTK_TYPE_WIDGET);
- list_signals[UNSELECT_CHILD] =
- gtk_signal_new (I_("unselect-child"),
- GTK_RUN_FIRST,
- GTK_CLASS_TYPE (object_class),
- GTK_SIGNAL_OFFSET (GtkListClass, unselect_child),
- _gtk_marshal_VOID__OBJECT,
- GTK_TYPE_NONE, 1,
- GTK_TYPE_WIDGET);
-
- gtk_object_add_arg_type ("GtkList::selection-mode",
- GTK_TYPE_SELECTION_MODE,
- GTK_ARG_READWRITE | G_PARAM_STATIC_NAME,
- ARG_SELECTION_MODE);
-}
-
-static void
-gtk_list_init (GtkList *list)
-{
- list->children = NULL;
- list->selection = NULL;
-
- list->undo_selection = NULL;
- list->undo_unselection = NULL;
-
- list->last_focus_child = NULL;
- list->undo_focus_child = NULL;
-
- list->htimer = 0;
- list->vtimer = 0;
-
- list->anchor = -1;
- list->drag_pos = -1;
- list->anchor_state = GTK_STATE_SELECTED;
-
- list->selection_mode = GTK_SELECTION_SINGLE;
- list->drag_selection = FALSE;
- list->add_mode = FALSE;
-}
-
-static void
-gtk_list_set_arg (GtkObject *object,
- GtkArg *arg,
- guint arg_id)
-{
- GtkList *list = GTK_LIST (object);
-
- switch (arg_id)
- {
- case ARG_SELECTION_MODE:
- gtk_list_set_selection_mode (list, GTK_VALUE_ENUM (*arg));
- break;
- }
-}
-
-static void
-gtk_list_get_arg (GtkObject *object,
- GtkArg *arg,
- guint arg_id)
-{
- GtkList *list = GTK_LIST (object);
-
- switch (arg_id)
- {
- case ARG_SELECTION_MODE:
- GTK_VALUE_ENUM (*arg) = list->selection_mode;
- break;
- default:
- arg->type = GTK_TYPE_INVALID;
- break;
- }
-}
-
-GtkWidget*
-gtk_list_new (void)
-{
- return GTK_WIDGET (gtk_type_new (GTK_TYPE_LIST));
-}
-
-
-/* Private GtkObject Methods :
- *
- * gtk_list_dispose
- */
-static void
-gtk_list_dispose (GObject *object)
-{
- gtk_list_clear_items (GTK_LIST (object), 0, -1);
-
- G_OBJECT_CLASS (parent_class)->dispose (object);
-}
-
-
-/* Private GtkWidget Methods :
- *
- * gtk_list_size_request
- * gtk_list_size_allocate
- * gtk_list_realize
- * gtk_list_unmap
- * gtk_list_motion_notify
- * gtk_list_button_press
- * gtk_list_button_release
- */
-static void
-gtk_list_size_request (GtkWidget *widget,
- GtkRequisition *requisition)
-{
- GtkList *list = GTK_LIST (widget);
- GtkWidget *child;
- GList *children;
-
- requisition->width = 0;
- requisition->height = 0;
-
- children = list->children;
- while (children)
- {
- child = children->data;
- children = children->next;
-
- if (gtk_widget_get_visible (child))
- {
- GtkRequisition child_requisition;
-
- gtk_widget_size_request (child, &child_requisition);
-
- requisition->width = MAX (requisition->width,
- child_requisition.width);
- requisition->height += child_requisition.height;
- }
- }
-
- requisition->width += GTK_CONTAINER (list)->border_width * 2;
- requisition->height += GTK_CONTAINER (list)->border_width * 2;
-
- requisition->width = MAX (requisition->width, 1);
- requisition->height = MAX (requisition->height, 1);
-}
-
-static void
-gtk_list_size_allocate (GtkWidget *widget,
- GtkAllocation *allocation)
-{
- GtkList *list = GTK_LIST (widget);
- GtkWidget *child;
- GtkAllocation child_allocation;
- GList *children;
-
- widget->allocation = *allocation;
- if (gtk_widget_get_realized (widget))
- gdk_window_move_resize (widget->window,
- allocation->x, allocation->y,
- allocation->width, allocation->height);
-
- if (list->children)
- {
- child_allocation.x = GTK_CONTAINER (list)->border_width;
- child_allocation.y = GTK_CONTAINER (list)->border_width;
- child_allocation.width = MAX (1, (gint)allocation->width -
- child_allocation.x * 2);
-
- children = list->children;
-
- while (children)
- {
- child = children->data;
- children = children->next;
-
- if (gtk_widget_get_visible (child))
- {
- GtkRequisition child_requisition;
- gtk_widget_get_child_requisition (child, &child_requisition);
-
- child_allocation.height = child_requisition.height;
-
- gtk_widget_size_allocate (child, &child_allocation);
-
- child_allocation.y += child_allocation.height;
- }
- }
- }
-}
-
-static void
-gtk_list_realize (GtkWidget *widget)
-{
- GdkWindowAttr attributes;
- gint attributes_mask;
-
- gtk_widget_set_realized (widget, TRUE);
-
- attributes.window_type = GDK_WINDOW_CHILD;
- attributes.x = widget->allocation.x;
- attributes.y = widget->allocation.y;
- attributes.width = widget->allocation.width;
- attributes.height = widget->allocation.height;
- attributes.wclass = GDK_INPUT_OUTPUT;
- attributes.visual = gtk_widget_get_visual (widget);
- attributes.colormap = gtk_widget_get_colormap (widget);
- attributes.event_mask = gtk_widget_get_events (widget) | GDK_EXPOSURE_MASK;
-
- 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]);
-}
-
-static gboolean
-list_has_grab (GtkList *list)
-{
- return (GTK_WIDGET_HAS_GRAB (list) &&
- gdk_display_pointer_is_grabbed (gtk_widget_get_display (GTK_WIDGET (list))));
-
-}
-
-static void
-gtk_list_unmap (GtkWidget *widget)
-{
- GtkList *list = GTK_LIST (widget);
-
- if (!gtk_widget_get_mapped (widget))
- return;
-
- gtk_widget_set_mapped (widget, FALSE);
-
- if (list_has_grab (list))
- {
- gtk_list_end_drag_selection (list);
-
- if (list->anchor != -1 && list->selection_mode == GTK_SELECTION_MULTIPLE)
- gtk_list_end_selection (list);
- }
-
- gdk_window_hide (widget->window);
-}
-
-static gint
-gtk_list_motion_notify (GtkWidget *widget,
- GdkEventMotion *event)
-{
- GtkList *list = GTK_LIST (widget);
- GtkWidget *item = NULL;
- GtkAdjustment *adj;
- GtkContainer *container;
- GList *work;
- gint x;
- gint y;
- gint row = -1;
- gint focus_row = 0;
- gint length = 0;
-
- if (!list->drag_selection || !list->children)
- return FALSE;
-
- container = GTK_CONTAINER (widget);
-
- if (event->is_hint || event->window != widget->window)
- gdk_window_get_pointer (widget->window, &x, &y, NULL);
- else
- {
- x = event->x;
- y = event->y;
- }
-
- adj = gtk_object_get_data_by_id (GTK_OBJECT (list), hadjustment_key_id);
-
- /* horizontal autoscrolling */
- if (adj && widget->allocation.width > adj->page_size &&
- (x < adj->value || x >= adj->value + adj->page_size))
- {
- if (list->htimer == 0)
- {
- list->htimer = gdk_threads_add_timeout
- (SCROLL_TIME, (GSourceFunc) gtk_list_horizontal_timeout, widget);
-
- if (!((x < adj->value && adj->value <= 0) ||
- (x > adj->value + adj->page_size &&
- adj->value >= adj->upper - adj->page_size)))
- {
- gdouble value;
-
- if (x < adj->value)
- value = adj->value + (x - adj->value) / 2 - 1;
- else
- value = adj->value + 1 + (x - adj->value - adj->page_size) / 2;
-
- gtk_adjustment_set_value (adj,
- CLAMP (value, 0.0,
- adj->upper - adj->page_size));
- }
- }
- else
- return FALSE;
- }
-
-
- /* vertical autoscrolling */
- for (work = list->children; work; length++, work = work->next)
- {
- if (row < 0)
- {
- item = GTK_WIDGET (work->data);
- if (item->allocation.y > y ||
- (item->allocation.y <= y &&
- item->allocation.y + item->allocation.height > y))
- row = length;
- }
-
- if (work->data == container->focus_child)
- focus_row = length;
- }
-
- if (row < 0)
- row = length - 1;
-
- if (list->vtimer != 0)
- return FALSE;
-
- if (!((y < 0 && focus_row == 0) ||
- (y > widget->allocation.height && focus_row >= length - 1)))
- list->vtimer = gdk_threads_add_timeout (SCROLL_TIME,
- (GSourceFunc) gtk_list_vertical_timeout,
- list);
-
- if (row != focus_row)
- gtk_widget_grab_focus (item);
-
- switch (list->selection_mode)
- {
- case GTK_SELECTION_BROWSE:
- gtk_list_select_child (list, item);
- break;
- case GTK_SELECTION_MULTIPLE:
- gtk_list_update_extended_selection (list, row);
- break;
- default:
- break;
- }
-
- return FALSE;
-}
-
-static gint
-gtk_list_button_press (GtkWidget *widget,
- GdkEventButton *event)
-{
- GtkList *list = GTK_LIST (widget);
- GtkWidget *item;
-
- if (event->button != 1)
- return FALSE;
-
- item = gtk_get_event_widget ((GdkEvent*) event);
-
- while (item && !GTK_IS_LIST_ITEM (item))
- item = item->parent;
-
- if (item && (item->parent == widget))
- {
- gint last_focus_row;
- gint focus_row;
-
- if (event->type == GDK_BUTTON_PRESS)
- {
- gtk_grab_add (widget);
- list->drag_selection = TRUE;
- }
- else if (list_has_grab (list))
- gtk_list_end_drag_selection (list);
-
- if (!gtk_widget_has_focus(item))
- gtk_widget_grab_focus (item);
-
- if (list->add_mode)
- {
- list->add_mode = FALSE;
- gtk_widget_queue_draw (item);
- }
-
- switch (list->selection_mode)
- {
- case GTK_SELECTION_SINGLE:
- if (event->type != GDK_BUTTON_PRESS)
- gtk_list_select_child (list, item);
- else
- list->undo_focus_child = item;
- break;
-
- case GTK_SELECTION_BROWSE:
- break;
-
- case GTK_SELECTION_MULTIPLE:
- focus_row = g_list_index (list->children, item);
-
- if (list->last_focus_child)
- last_focus_row = g_list_index (list->children,
- list->last_focus_child);
- else
- {
- last_focus_row = focus_row;
- list->last_focus_child = item;
- }
-
- if (event->type != GDK_BUTTON_PRESS)
- {
- if (list->anchor >= 0)
- {
- gtk_list_update_extended_selection (list, focus_row);
- gtk_list_end_selection (list);
- }
- gtk_list_select_child (list, item);
- break;
- }
-
- if (event->state & GDK_CONTROL_MASK)
- {
- if (event->state & GDK_SHIFT_MASK)
- {
- if (list->anchor < 0)
- {
- g_list_free (list->undo_selection);
- g_list_free (list->undo_unselection);
- list->undo_selection = NULL;
- list->undo_unselection = NULL;
-
- list->anchor = last_focus_row;
- list->drag_pos = last_focus_row;
- list->undo_focus_child = list->last_focus_child;
- }
- gtk_list_update_extended_selection (list, focus_row);
- }
- else
- {
- if (list->anchor < 0)
- gtk_list_set_anchor (list, TRUE,
- focus_row, list->last_focus_child);
- else
- gtk_list_update_extended_selection (list, focus_row);
- }
- break;
- }
-
- if (event->state & GDK_SHIFT_MASK)
- {
- gtk_list_set_anchor (list, FALSE,
- last_focus_row, list->last_focus_child);
- gtk_list_update_extended_selection (list, focus_row);
- break;
- }
-
- if (list->anchor < 0)
- gtk_list_set_anchor (list, FALSE, focus_row,
- list->last_focus_child);
- else
- gtk_list_update_extended_selection (list, focus_row);
- break;
-
- default:
- break;
- }
-
- return TRUE;
- }
-
- return FALSE;
-}
-
-static gint
-gtk_list_button_release (GtkWidget *widget,
- GdkEventButton *event)
-{
- GtkList *list = GTK_LIST (widget);
- GtkWidget *item;
-
- /* we don't handle button 2 and 3 */
- if (event->button != 1)
- return FALSE;
-
- if (list->drag_selection)
- {
- gtk_list_end_drag_selection (list);
-
- switch (list->selection_mode)
- {
- case GTK_SELECTION_MULTIPLE:
- if (!(event->state & GDK_SHIFT_MASK))
- gtk_list_end_selection (list);
- break;
-
- case GTK_SELECTION_SINGLE:
-
- item = gtk_get_event_widget ((GdkEvent*) event);
-
- while (item && !GTK_IS_LIST_ITEM (item))
- item = item->parent;
-
- if (item && item->parent == widget)
- {
- if (list->undo_focus_child == item)
- gtk_list_toggle_row (list, item);
- }
- list->undo_focus_child = NULL;
- break;
-
- default:
- break;
- }
-
- return TRUE;
- }
-
- return FALSE;
-}
-
-static void
-gtk_list_style_set (GtkWidget *widget,
- GtkStyle *previous_style)
-{
- GtkStyle *style;
-
- if (previous_style && gtk_widget_get_realized (widget))
- {
- style = gtk_widget_get_style (widget);
- gdk_window_set_background (gtk_widget_get_window (widget),
- &style->base[gtk_widget_get_state (widget)]);
- }
-}
-
-/* GtkContainer Methods :
- * gtk_list_add
- * gtk_list_remove
- * gtk_list_forall
- * gtk_list_child_type
- * gtk_list_set_focus_child
- * gtk_list_focus
- */
-static void
-gtk_list_add (GtkContainer *container,
- GtkWidget *widget)
-{
- GList *item_list;
-
- g_return_if_fail (GTK_IS_LIST_ITEM (widget));
-
- item_list = g_list_alloc ();
- item_list->data = widget;
-
- gtk_list_append_items (GTK_LIST (container), item_list);
-}
-
-static void
-gtk_list_remove (GtkContainer *container,
- GtkWidget *widget)
-{
- GList *item_list;
-
- g_return_if_fail (container == GTK_CONTAINER (widget->parent));
-
- item_list = g_list_alloc ();
- item_list->data = widget;
-
- gtk_list_remove_items (GTK_LIST (container), item_list);
-
- g_list_free (item_list);
-}
-
-static void
-gtk_list_forall (GtkContainer *container,
- gboolean include_internals,
- GtkCallback callback,
- gpointer callback_data)
-{
- GtkList *list = GTK_LIST (container);
- GtkWidget *child;
- GList *children;
-
- children = list->children;
-
- while (children)
- {
- child = children->data;
- children = children->next;
-
- (* callback) (child, callback_data);
- }
-}
-
-static GtkType
-gtk_list_child_type (GtkContainer *container)
-{
- return GTK_TYPE_LIST_ITEM;
-}
-
-static void
-gtk_list_set_focus_child (GtkContainer *container,
- GtkWidget *child)
-{
- GtkList *list;
-
- g_return_if_fail (GTK_IS_LIST (container));
-
- if (child)
- g_return_if_fail (GTK_IS_WIDGET (child));
-
- list = GTK_LIST (container);
-
- if (child != container->focus_child)
- {
- if (container->focus_child)
- {
- list->last_focus_child = container->focus_child;
- g_object_unref (container->focus_child);
- }
- container->focus_child = child;
- if (container->focus_child)
- g_object_ref (container->focus_child);
- }
-
- /* check for v adjustment */
- if (container->focus_child)
- {
- GtkAdjustment *adjustment;
-
- adjustment = gtk_object_get_data_by_id (GTK_OBJECT (container),
- vadjustment_key_id);
- if (adjustment)
- gtk_adjustment_clamp_page (adjustment,
- container->focus_child->allocation.y,
- (container->focus_child->allocation.y +
- container->focus_child->allocation.height));
- switch (list->selection_mode)
- {
- case GTK_SELECTION_BROWSE:
- gtk_list_select_child (list, child);
- break;
- case GTK_SELECTION_MULTIPLE:
- if (!list->last_focus_child && !list->add_mode)
- {
- list->undo_focus_child = list->last_focus_child;
- gtk_list_unselect_all (list);
- gtk_list_select_child (list, child);
- }
- break;
- default:
- break;
- }
- }
-}
-
-static gboolean
-gtk_list_focus (GtkWidget *widget,
- GtkDirectionType direction)
-{
- gint return_val = FALSE;
- GtkContainer *container;
-
- container = GTK_CONTAINER (widget);
-
- if (container->focus_child == NULL ||
- !gtk_widget_has_focus (container->focus_child))
- {
- if (GTK_LIST (container)->last_focus_child)
- gtk_container_set_focus_child
- (container, GTK_LIST (container)->last_focus_child);
-
- if (GTK_WIDGET_CLASS (parent_class)->focus)
- return_val = GTK_WIDGET_CLASS (parent_class)->focus (widget,
- direction);
- }
-
- if (!return_val)
- {
- GtkList *list;
-
- list = GTK_LIST (container);
- if (list->selection_mode == GTK_SELECTION_MULTIPLE && list->anchor >= 0)
- gtk_list_end_selection (list);
-
- if (container->focus_child)
- list->last_focus_child = container->focus_child;
- }
-
- return return_val;
-}
-
-
-/* Public GtkList Methods :
- *
- * gtk_list_insert_items
- * gtk_list_append_items
- * gtk_list_prepend_items
- * gtk_list_remove_items
- * gtk_list_remove_items_no_unref
- * gtk_list_clear_items
- *
- * gtk_list_child_position
- */
-void
-gtk_list_insert_items (GtkList *list,
- GList *items,
- gint position)
-{
- GtkWidget *widget;
- GList *tmp_list;
- GList *last;
- gint nchildren;
-
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (!items)
- return;
-
- gtk_list_end_drag_selection (list);
- if (list->selection_mode == GTK_SELECTION_MULTIPLE && list->anchor >= 0)
- gtk_list_end_selection (list);
-
- tmp_list = items;
- while (tmp_list)
- {
- widget = tmp_list->data;
- tmp_list = tmp_list->next;
-
- gtk_widget_set_parent (widget, GTK_WIDGET (list));
- gtk_signal_connect (GTK_OBJECT (widget), "drag-begin",
- G_CALLBACK (gtk_list_signal_drag_begin),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "toggle-focus-row",
- G_CALLBACK (gtk_list_signal_toggle_focus_row),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "select-all",
- G_CALLBACK (gtk_list_signal_select_all),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "unselect-all",
- G_CALLBACK (gtk_list_signal_unselect_all),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "undo-selection",
- G_CALLBACK (gtk_list_signal_undo_selection),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "start-selection",
- G_CALLBACK (gtk_list_signal_start_selection),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "end-selection",
- G_CALLBACK (gtk_list_signal_end_selection),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "extend-selection",
- G_CALLBACK (gtk_list_signal_extend_selection),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "scroll-horizontal",
- G_CALLBACK (gtk_list_signal_scroll_horizontal),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "scroll-vertical",
- G_CALLBACK (gtk_list_signal_scroll_vertical),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "toggle-add-mode",
- G_CALLBACK (gtk_list_signal_toggle_add_mode),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "select",
- G_CALLBACK (gtk_list_signal_item_select),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "deselect",
- G_CALLBACK (gtk_list_signal_item_deselect),
- list);
- gtk_signal_connect (GTK_OBJECT (widget), "toggle",
- G_CALLBACK (gtk_list_signal_item_toggle),
- list);
- }
-
-
- nchildren = g_list_length (list->children);
- if ((position < 0) || (position > nchildren))
- position = nchildren;
-
- if (position == nchildren)
- {
- if (list->children)
- {
- tmp_list = g_list_last (list->children);
- tmp_list->next = items;
- items->prev = tmp_list;
- }
- else
- {
- list->children = items;
- }
- }
- else
- {
- tmp_list = g_list_nth (list->children, position);
- last = g_list_last (items);
-
- if (tmp_list->prev)
- tmp_list->prev->next = items;
- last->next = tmp_list;
- items->prev = tmp_list->prev;
- tmp_list->prev = last;
-
- if (tmp_list == list->children)
- list->children = items;
- }
-
- if (list->children && !list->selection &&
- (list->selection_mode == GTK_SELECTION_BROWSE))
- {
- widget = list->children->data;
- gtk_list_select_child (list, widget);
- }
-}
-
-void
-gtk_list_append_items (GtkList *list,
- GList *items)
-{
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_insert_items (list, items, -1);
-}
-
-void
-gtk_list_prepend_items (GtkList *list,
- GList *items)
-{
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_insert_items (list, items, 0);
-}
-
-void
-gtk_list_remove_items (GtkList *list,
- GList *items)
-{
- gtk_list_remove_items_internal (list, items, FALSE);
-}
-
-void
-gtk_list_remove_items_no_unref (GtkList *list,
- GList *items)
-{
- gtk_list_remove_items_internal (list, items, TRUE);
-}
-
-void
-gtk_list_clear_items (GtkList *list,
- gint start,
- gint end)
-{
- GtkContainer *container;
- GtkWidget *widget;
- GtkWidget *new_focus_child = NULL;
- GList *start_list;
- GList *end_list;
- GList *tmp_list;
- guint nchildren;
- gboolean grab_focus = FALSE;
-
- g_return_if_fail (GTK_IS_LIST (list));
-
- nchildren = g_list_length (list->children);
-
- if (nchildren == 0)
- return;
-
- if ((end < 0) || (end > nchildren))
- end = nchildren;
-
- if (start >= end)
- return;
-
- container = GTK_CONTAINER (list);
-
- gtk_list_end_drag_selection (list);
- if (list->selection_mode == GTK_SELECTION_MULTIPLE)
- {
- if (list->anchor >= 0)
- gtk_list_end_selection (list);
-
- gtk_list_reset_extended_selection (list);
- }
-
- start_list = g_list_nth (list->children, start);
- end_list = g_list_nth (list->children, end);
-
- if (start_list->prev)
- start_list->prev->next = end_list;
- if (end_list && end_list->prev)
- end_list->prev->next = NULL;
- if (end_list)
- end_list->prev = start_list->prev;
- if (start_list == list->children)
- list->children = end_list;
-
- if (container->focus_child)
- {
- if (g_list_find (start_list, container->focus_child))
- {
- if (start_list->prev)
- new_focus_child = start_list->prev->data;
- else if (list->children)
- new_focus_child = list->children->data;
-
- if (gtk_widget_has_focus (container->focus_child))
- grab_focus = TRUE;
- }
- }
-
- tmp_list = start_list;
- while (tmp_list)
- {
- widget = tmp_list->data;
- tmp_list = tmp_list->next;
-
- g_object_ref (widget);
-
- if (widget->state == GTK_STATE_SELECTED)
- gtk_list_unselect_child (list, widget);
-
- gtk_signal_disconnect_by_data (GTK_OBJECT (widget), (gpointer) list);
- gtk_widget_unparent (widget);
-
- if (widget == list->undo_focus_child)
- list->undo_focus_child = NULL;
- if (widget == list->last_focus_child)
- list->last_focus_child = NULL;
-
- g_object_unref (widget);
- }
-
- g_list_free (start_list);
-
- if (new_focus_child)
- {
- if (grab_focus)
- gtk_widget_grab_focus (new_focus_child);
- else if (container->focus_child)
- gtk_container_set_focus_child (container, new_focus_child);
-
- if ((list->selection_mode == GTK_SELECTION_BROWSE ||
- list->selection_mode == GTK_SELECTION_MULTIPLE) && !list->selection)
- {
- list->last_focus_child = new_focus_child;
- gtk_list_select_child (list, new_focus_child);
- }
- }
-
- if (gtk_widget_get_visible (GTK_WIDGET (list)))
- gtk_widget_queue_resize (GTK_WIDGET (list));
-}
-
-gint
-gtk_list_child_position (GtkList *list,
- GtkWidget *child)
-{
- GList *children;
- gint pos;
-
- g_return_val_if_fail (GTK_IS_LIST (list), -1);
- g_return_val_if_fail (child != NULL, -1);
-
- pos = 0;
- children = list->children;
-
- while (children)
- {
- if (child == GTK_WIDGET (children->data))
- return pos;
-
- pos += 1;
- children = children->next;
- }
-
- return -1;
-}
-
-
-/* Private GtkList Insert/Remove Item Functions:
- *
- * gtk_list_remove_items_internal
- */
-static void
-gtk_list_remove_items_internal (GtkList *list,
- GList *items,
- gboolean no_unref)
-{
- GtkWidget *widget;
- GtkWidget *new_focus_child;
- GtkWidget *old_focus_child;
- GtkContainer *container;
- GList *tmp_list;
- GList *work;
- gboolean grab_focus = FALSE;
-
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (!items)
- return;
-
- container = GTK_CONTAINER (list);
-
- gtk_list_end_drag_selection (list);
- if (list->selection_mode == GTK_SELECTION_MULTIPLE)
- {
- if (list->anchor >= 0)
- gtk_list_end_selection (list);
-
- gtk_list_reset_extended_selection (list);
- }
-
- tmp_list = items;
- while (tmp_list)
- {
- widget = tmp_list->data;
- tmp_list = tmp_list->next;
-
- if (widget->state == GTK_STATE_SELECTED)
- gtk_list_unselect_child (list, widget);
- }
-
- if (container->focus_child)
- {
- old_focus_child = new_focus_child = container->focus_child;
- if (gtk_widget_has_focus (container->focus_child))
- grab_focus = TRUE;
- }
- else
- old_focus_child = new_focus_child = list->last_focus_child;
-
- tmp_list = items;
- while (tmp_list)
- {
- widget = tmp_list->data;
- tmp_list = tmp_list->next;
-
- g_object_ref (widget);
- if (no_unref)
- g_object_ref (widget);
-
- if (widget == new_focus_child)
- {
- work = g_list_find (list->children, widget);
-
- if (work)
- {
- if (work->next)
- new_focus_child = work->next->data;
- else if (list->children != work && work->prev)
- new_focus_child = work->prev->data;
- else
- new_focus_child = NULL;
- }
- }
-
- gtk_signal_disconnect_by_data (GTK_OBJECT (widget), (gpointer) list);
- list->children = g_list_remove (list->children, widget);
- gtk_widget_unparent (widget);
-
- if (widget == list->undo_focus_child)
- list->undo_focus_child = NULL;
- if (widget == list->last_focus_child)
- list->last_focus_child = NULL;
-
- g_object_unref (widget);
- }
-
- if (new_focus_child && new_focus_child != old_focus_child)
- {
- if (grab_focus)
- gtk_widget_grab_focus (new_focus_child);
- else if (container->focus_child)
- gtk_container_set_focus_child (container, new_focus_child);
-
- if (list->selection_mode == GTK_SELECTION_BROWSE && !list->selection)
- {
- list->last_focus_child = new_focus_child;
- gtk_list_select_child (list, new_focus_child);
- }
- }
-
- if (gtk_widget_get_visible (GTK_WIDGET (list)))
- gtk_widget_queue_resize (GTK_WIDGET (list));
-}
-
-
-/* Public GtkList Selection Methods :
- *
- * gtk_list_set_selection_mode
- * gtk_list_select_item
- * gtk_list_unselect_item
- * gtk_list_select_child
- * gtk_list_unselect_child
- * gtk_list_select_all
- * gtk_list_unselect_all
- * gtk_list_extend_selection
- * gtk_list_end_drag_selection
- * gtk_list_start_selection
- * gtk_list_end_selection
- * gtk_list_toggle_row
- * gtk_list_toggle_focus_row
- * gtk_list_toggle_add_mode
- * gtk_list_undo_selection
- */
-void
-gtk_list_set_selection_mode (GtkList *list,
- GtkSelectionMode mode)
-{
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (list->selection_mode == mode)
- return;
-
- list->selection_mode = mode;
-
- switch (mode)
- {
- case GTK_SELECTION_SINGLE:
- case GTK_SELECTION_BROWSE:
- gtk_list_unselect_all (list);
- break;
- default:
- break;
- }
-}
-
-void
-gtk_list_select_item (GtkList *list,
- gint item)
-{
- GList *tmp_list;
-
- g_return_if_fail (GTK_IS_LIST (list));
-
- tmp_list = g_list_nth (list->children, item);
- if (tmp_list)
- gtk_list_select_child (list, GTK_WIDGET (tmp_list->data));
-}
-
-void
-gtk_list_unselect_item (GtkList *list,
- gint item)
-{
- GList *tmp_list;
-
- g_return_if_fail (GTK_IS_LIST (list));
-
- tmp_list = g_list_nth (list->children, item);
- if (tmp_list)
- gtk_list_unselect_child (list, GTK_WIDGET (tmp_list->data));
-}
-
-void
-gtk_list_select_child (GtkList *list,
- GtkWidget *child)
-{
- gtk_signal_emit (GTK_OBJECT (list), list_signals[SELECT_CHILD], child);
-}
-
-void
-gtk_list_unselect_child (GtkList *list,
- GtkWidget *child)
-{
- gtk_signal_emit (GTK_OBJECT (list), list_signals[UNSELECT_CHILD], child);
-}
-
-void
-gtk_list_select_all (GtkList *list)
-{
- GtkContainer *container;
-
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (!list->children)
- return;
-
- if (list_has_grab (list))
- gtk_list_end_drag_selection (list);
-
- if (list->selection_mode == GTK_SELECTION_MULTIPLE && list->anchor >= 0)
- gtk_list_end_selection (list);
-
- container = GTK_CONTAINER (list);
-
- switch (list->selection_mode)
- {
- case GTK_SELECTION_BROWSE:
- if (container->focus_child)
- {
- gtk_list_select_child (list, container->focus_child);
- return;
- }
- break;
- case GTK_SELECTION_MULTIPLE:
- g_list_free (list->undo_selection);
- g_list_free (list->undo_unselection);
- list->undo_selection = NULL;
- list->undo_unselection = NULL;
-
- if (list->children &&
- gtk_widget_get_state (list->children->data) != GTK_STATE_SELECTED)
- gtk_list_fake_toggle_row (list, GTK_WIDGET (list->children->data));
-
- list->anchor_state = GTK_STATE_SELECTED;
- list->anchor = 0;
- list->drag_pos = 0;
- list->undo_focus_child = container->focus_child;
- gtk_list_update_extended_selection (list, g_list_length(list->children));
- gtk_list_end_selection (list);
- return;
- default:
- break;
- }
-}
-
-void
-gtk_list_unselect_all (GtkList *list)
-{
- GtkContainer *container;
- GtkWidget *item;
- GList *work;
-
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (!list->children)
- return;
-
- if (list_has_grab (list))
- gtk_list_end_drag_selection (list);
-
- if (list->selection_mode == GTK_SELECTION_MULTIPLE && list->anchor >= 0)
- gtk_list_end_selection (list);
-
- container = GTK_CONTAINER (list);
-
- switch (list->selection_mode)
- {
- case GTK_SELECTION_BROWSE:
- if (container->focus_child)
- {
- gtk_list_select_child (list, container->focus_child);
- return;
- }
- break;
- case GTK_SELECTION_MULTIPLE:
- gtk_list_reset_extended_selection (list);
- break;
- default:
- break;
- }
-
- work = list->selection;
-
- while (work)
- {
- item = work->data;
- work = work->next;
- gtk_list_unselect_child (list, item);
- }
-}
-
-void
-gtk_list_extend_selection (GtkList *list,
- GtkScrollType scroll_type,
- gfloat position,
- gboolean auto_start_selection)
-{
- GtkContainer *container;
-
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (list_has_grab (list) ||
- list->selection_mode != GTK_SELECTION_MULTIPLE)
- return;
-
- container = GTK_CONTAINER (list);
-
- if (auto_start_selection)
- {
- gint focus_row;
-
- focus_row = g_list_index (list->children, container->focus_child);
- gtk_list_set_anchor (list, list->add_mode, focus_row,
- container->focus_child);
- }
- else if (list->anchor < 0)
- return;
-
- gtk_list_move_focus_child (list, scroll_type, position);
- gtk_list_update_extended_selection
- (list, g_list_index (list->children, container->focus_child));
-}
-
-void
-gtk_list_end_drag_selection (GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST (list));
-
- list->drag_selection = FALSE;
- if (GTK_WIDGET_HAS_GRAB (list))
- gtk_grab_remove (GTK_WIDGET (list));
-
- if (list->htimer)
- {
- g_source_remove (list->htimer);
- list->htimer = 0;
- }
- if (list->vtimer)
- {
- g_source_remove (list->vtimer);
- list->vtimer = 0;
- }
-}
-
-void
-gtk_list_start_selection (GtkList *list)
-{
- GtkContainer *container;
- gint focus_row;
-
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (list_has_grab (list))
- return;
-
- container = GTK_CONTAINER (list);
-
- if ((focus_row = g_list_index (list->selection, container->focus_child))
- >= 0)
- gtk_list_set_anchor (list, list->add_mode,
- focus_row, container->focus_child);
-}
-
-void
-gtk_list_end_selection (GtkList *list)
-{
- gint i;
- gint e;
- gboolean top_down;
- GList *work;
- GtkWidget *item;
- gint item_index;
-
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (list_has_grab (list) || list->anchor < 0)
- return;
-
- i = MIN (list->anchor, list->drag_pos);
- e = MAX (list->anchor, list->drag_pos);
-
- top_down = (list->anchor < list->drag_pos);
-
- list->anchor = -1;
- list->drag_pos = -1;
-
- if (list->undo_selection)
- {
- work = list->selection;
- list->selection = list->undo_selection;
- list->undo_selection = work;
- work = list->selection;
- while (work)
- {
- item = work->data;
- work = work->next;
- item_index = g_list_index (list->children, item);
- if (item_index < i || item_index > e)
- {
- gtk_widget_set_state (item, GTK_STATE_SELECTED);
- gtk_list_unselect_child (list, item);
- list->undo_selection = g_list_prepend (list->undo_selection,
- item);
- }
- }
- }
-
- if (top_down)
- {
- for (work = g_list_nth (list->children, i); i <= e;
- i++, work = work->next)
- {
- item = work->data;
- if (g_list_find (list->selection, item))
- {
- if (item->state == GTK_STATE_NORMAL)
- {
- gtk_widget_set_state (item, GTK_STATE_SELECTED);
- gtk_list_unselect_child (list, item);
- list->undo_selection = g_list_prepend (list->undo_selection,
- item);
- }
- }
- else if (item->state == GTK_STATE_SELECTED)
- {
- gtk_widget_set_state (item, GTK_STATE_NORMAL);
- list->undo_unselection = g_list_prepend (list->undo_unselection,
- item);
- }
- }
- }
- else
- {
- for (work = g_list_nth (list->children, e); i <= e;
- e--, work = work->prev)
- {
- item = work->data;
- if (g_list_find (list->selection, item))
- {
- if (item->state == GTK_STATE_NORMAL)
- {
- gtk_widget_set_state (item, GTK_STATE_SELECTED);
- gtk_list_unselect_child (list, item);
- list->undo_selection = g_list_prepend (list->undo_selection,
- item);
- }
- }
- else if (item->state == GTK_STATE_SELECTED)
- {
- gtk_widget_set_state (item, GTK_STATE_NORMAL);
- list->undo_unselection = g_list_prepend (list->undo_unselection,
- item);
- }
- }
- }
-
- for (work = g_list_reverse (list->undo_unselection); work; work = work->next)
- gtk_list_select_child (list, GTK_WIDGET (work->data));
-
-
-}
-
-void
-gtk_list_toggle_row (GtkList *list,
- GtkWidget *item)
-{
- g_return_if_fail (GTK_IS_LIST (list));
- g_return_if_fail (GTK_IS_LIST_ITEM (item));
-
- switch (list->selection_mode)
- {
- case GTK_SELECTION_MULTIPLE:
- case GTK_SELECTION_SINGLE:
- if (item->state == GTK_STATE_SELECTED)
- {
- gtk_list_unselect_child (list, item);
- return;
- }
- case GTK_SELECTION_BROWSE:
- gtk_list_select_child (list, item);
- break;
- }
-}
-
-void
-gtk_list_toggle_focus_row (GtkList *list)
-{
- GtkContainer *container;
- gint focus_row;
-
- g_return_if_fail (list != 0);
- g_return_if_fail (GTK_IS_LIST (list));
-
- container = GTK_CONTAINER (list);
-
- if (list_has_grab (list) || !container->focus_child)
- return;
-
- switch (list->selection_mode)
- {
- case GTK_SELECTION_SINGLE:
- gtk_list_toggle_row (list, container->focus_child);
- break;
- case GTK_SELECTION_MULTIPLE:
- if ((focus_row = g_list_index (list->children, container->focus_child))
- < 0)
- return;
-
- g_list_free (list->undo_selection);
- g_list_free (list->undo_unselection);
- list->undo_selection = NULL;
- list->undo_unselection = NULL;
-
- list->anchor = focus_row;
- list->drag_pos = focus_row;
- list->undo_focus_child = container->focus_child;
-
- if (list->add_mode)
- gtk_list_fake_toggle_row (list, container->focus_child);
- else
- gtk_list_fake_unselect_all (list, container->focus_child);
-
- gtk_list_end_selection (list);
- break;
- default:
- break;
- }
-}
-
-void
-gtk_list_toggle_add_mode (GtkList *list)
-{
- GtkContainer *container;
-
- g_return_if_fail (list != 0);
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (list_has_grab (list) ||
- list->selection_mode != GTK_SELECTION_MULTIPLE)
- return;
-
- container = GTK_CONTAINER (list);
-
- if (list->add_mode)
- {
- list->add_mode = FALSE;
- list->anchor_state = GTK_STATE_SELECTED;
- }
- else
- list->add_mode = TRUE;
-
- if (container->focus_child)
- gtk_widget_queue_draw (container->focus_child);
-}
-
-void
-gtk_list_undo_selection (GtkList *list)
-{
- GList *work;
-
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (list->selection_mode != GTK_SELECTION_MULTIPLE ||
- list_has_grab (list))
- return;
-
- if (list->anchor >= 0)
- gtk_list_end_selection (list);
-
- if (!(list->undo_selection || list->undo_unselection))
- {
- gtk_list_unselect_all (list);
- return;
- }
-
- for (work = list->undo_selection; work; work = work->next)
- gtk_list_select_child (list, GTK_WIDGET (work->data));
-
- for (work = list->undo_unselection; work; work = work->next)
- gtk_list_unselect_child (list, GTK_WIDGET (work->data));
-
- if (list->undo_focus_child)
- {
- GtkContainer *container;
-
- container = GTK_CONTAINER (list);
-
- if (container->focus_child &&
- gtk_widget_has_focus (container->focus_child))
- gtk_widget_grab_focus (list->undo_focus_child);
- else
- gtk_container_set_focus_child (container, list->undo_focus_child);
- }
-
- list->undo_focus_child = NULL;
-
- g_list_free (list->undo_selection);
- g_list_free (list->undo_unselection);
- list->undo_selection = NULL;
- list->undo_unselection = NULL;
-}
-
-
-/* Private GtkList Selection Methods :
- *
- * gtk_real_list_select_child
- * gtk_real_list_unselect_child
- */
-static void
-gtk_real_list_select_child (GtkList *list,
- GtkWidget *child)
-{
- g_return_if_fail (GTK_IS_LIST (list));
- g_return_if_fail (GTK_IS_LIST_ITEM (child));
-
- switch (child->state)
- {
- case GTK_STATE_SELECTED:
- case GTK_STATE_INSENSITIVE:
- break;
- default:
- gtk_list_item_select (GTK_LIST_ITEM (child));
- break;
- }
-}
-
-static void
-gtk_real_list_unselect_child (GtkList *list,
- GtkWidget *child)
-{
- g_return_if_fail (GTK_IS_LIST (list));
- g_return_if_fail (GTK_IS_LIST_ITEM (child));
-
- if (child->state == GTK_STATE_SELECTED)
- gtk_list_item_deselect (GTK_LIST_ITEM (child));
-}
-
-
-/* Private GtkList Selection Functions :
- *
- * gtk_list_set_anchor
- * gtk_list_fake_unselect_all
- * gtk_list_fake_toggle_row
- * gtk_list_update_extended_selection
- * gtk_list_reset_extended_selection
- */
-static void
-gtk_list_set_anchor (GtkList *list,
- gboolean add_mode,
- gint anchor,
- GtkWidget *undo_focus_child)
-{
- GList *work;
-
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (list->selection_mode != GTK_SELECTION_MULTIPLE || list->anchor >= 0)
- return;
-
- g_list_free (list->undo_selection);
- g_list_free (list->undo_unselection);
- list->undo_selection = NULL;
- list->undo_unselection = NULL;
-
- if ((work = g_list_nth (list->children, anchor)))
- {
- if (add_mode)
- gtk_list_fake_toggle_row (list, GTK_WIDGET (work->data));
- else
- {
- gtk_list_fake_unselect_all (list, GTK_WIDGET (work->data));
- list->anchor_state = GTK_STATE_SELECTED;
- }
- }
-
- list->anchor = anchor;
- list->drag_pos = anchor;
- list->undo_focus_child = undo_focus_child;
-}
-
-static void
-gtk_list_fake_unselect_all (GtkList *list,
- GtkWidget *item)
-{
- GList *work;
-
- if (item && item->state == GTK_STATE_NORMAL)
- gtk_widget_set_state (item, GTK_STATE_SELECTED);
-
- list->undo_selection = list->selection;
- list->selection = NULL;
-
- for (work = list->undo_selection; work; work = work->next)
- if (work->data != item)
- gtk_widget_set_state (GTK_WIDGET (work->data), GTK_STATE_NORMAL);
-}
-
-static void
-gtk_list_fake_toggle_row (GtkList *list,
- GtkWidget *item)
-{
- if (!item)
- return;
-
- if (item->state == GTK_STATE_NORMAL)
- {
- list->anchor_state = GTK_STATE_SELECTED;
- gtk_widget_set_state (item, GTK_STATE_SELECTED);
- }
- else
- {
- list->anchor_state = GTK_STATE_NORMAL;
- gtk_widget_set_state (item, GTK_STATE_NORMAL);
- }
-}
-
-static void
-gtk_list_update_extended_selection (GtkList *list,
- gint row)
-{
- gint i;
- GList *work;
- gint s1 = -1;
- gint s2 = -1;
- gint e1 = -1;
- gint e2 = -1;
- gint length;
-
- if (row < 0)
- row = 0;
-
- length = g_list_length (list->children);
- if (row >= length)
- row = length - 1;
-
- if (list->selection_mode != GTK_SELECTION_MULTIPLE || !list->anchor < 0)
- return;
-
- /* extending downwards */
- if (row > list->drag_pos && list->anchor <= list->drag_pos)
- {
- s2 = list->drag_pos + 1;
- e2 = row;
- }
- /* extending upwards */
- else if (row < list->drag_pos && list->anchor >= list->drag_pos)
- {
- s2 = row;
- e2 = list->drag_pos - 1;
- }
- else if (row < list->drag_pos && list->anchor < list->drag_pos)
- {
- e1 = list->drag_pos;
- /* row and drag_pos on different sides of anchor :
- take back the selection between anchor and drag_pos,
- select between anchor and row */
- if (row < list->anchor)
- {
- s1 = list->anchor + 1;
- s2 = row;
- e2 = list->anchor - 1;
- }
- /* take back the selection between anchor and drag_pos */
- else
- s1 = row + 1;
- }
- else if (row > list->drag_pos && list->anchor > list->drag_pos)
- {
- s1 = list->drag_pos;
- /* row and drag_pos on different sides of anchor :
- take back the selection between anchor and drag_pos,
- select between anchor and row */
- if (row > list->anchor)
- {
- e1 = list->anchor - 1;
- s2 = list->anchor + 1;
- e2 = row;
- }
- /* take back the selection between anchor and drag_pos */
- else
- e1 = row - 1;
- }
-
- list->drag_pos = row;
-
- /* restore the elements between s1 and e1 */
- if (s1 >= 0)
- {
- for (i = s1, work = g_list_nth (list->children, i); i <= e1;
- i++, work = work->next)
- {
- if (g_list_find (list->selection, work->data))
- gtk_widget_set_state (GTK_WIDGET (work->data), GTK_STATE_SELECTED);
- else
- gtk_widget_set_state (GTK_WIDGET (work->data), GTK_STATE_NORMAL);
- }
- }
-
- /* extend the selection between s2 and e2 */
- if (s2 >= 0)
- {
- for (i = s2, work = g_list_nth (list->children, i); i <= e2;
- i++, work = work->next)
- if (GTK_WIDGET (work->data)->state != list->anchor_state)
- gtk_widget_set_state (GTK_WIDGET (work->data), list->anchor_state);
- }
-}
-
-static void
-gtk_list_reset_extended_selection (GtkList *list)
-{
- g_return_if_fail (list != 0);
- g_return_if_fail (GTK_IS_LIST (list));
-
- g_list_free (list->undo_selection);
- g_list_free (list->undo_unselection);
- list->undo_selection = NULL;
- list->undo_unselection = NULL;
-
- list->anchor = -1;
- list->drag_pos = -1;
- list->undo_focus_child = GTK_CONTAINER (list)->focus_child;
-}
-
-/* Public GtkList Scroll Methods :
- *
- * gtk_list_scroll_horizontal
- * gtk_list_scroll_vertical
- */
-void
-gtk_list_scroll_horizontal (GtkList *list,
- GtkScrollType scroll_type,
- gfloat position)
-{
- GtkAdjustment *adj;
-
- g_return_if_fail (list != 0);
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (list_has_grab (list))
- return;
-
- if (!(adj =
- gtk_object_get_data_by_id (GTK_OBJECT (list), hadjustment_key_id)))
- return;
-
- switch (scroll_type)
- {
- case GTK_SCROLL_STEP_UP:
- case GTK_SCROLL_STEP_BACKWARD:
- adj->value = CLAMP (adj->value - adj->step_increment, adj->lower,
- adj->upper - adj->page_size);
- break;
- case GTK_SCROLL_STEP_DOWN:
- case GTK_SCROLL_STEP_FORWARD:
- adj->value = CLAMP (adj->value + adj->step_increment, adj->lower,
- adj->upper - adj->page_size);
- break;
- case GTK_SCROLL_PAGE_UP:
- case GTK_SCROLL_PAGE_BACKWARD:
- adj->value = CLAMP (adj->value - adj->page_increment, adj->lower,
- adj->upper - adj->page_size);
- break;
- case GTK_SCROLL_PAGE_DOWN:
- case GTK_SCROLL_PAGE_FORWARD:
- adj->value = CLAMP (adj->value + adj->page_increment, adj->lower,
- adj->upper - adj->page_size);
- break;
- case GTK_SCROLL_JUMP:
- adj->value = CLAMP (adj->lower + (adj->upper - adj->lower) * position,
- adj->lower, adj->upper - adj->page_size);
- break;
- default:
- break;
- }
- gtk_adjustment_value_changed (adj);
-}
-
-void
-gtk_list_scroll_vertical (GtkList *list,
- GtkScrollType scroll_type,
- gfloat position)
-{
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (list_has_grab (list))
- return;
-
- if (list->selection_mode == GTK_SELECTION_MULTIPLE)
- {
- GtkContainer *container;
-
- if (list->anchor >= 0)
- return;
-
- container = GTK_CONTAINER (list);
- list->undo_focus_child = container->focus_child;
- gtk_list_move_focus_child (list, scroll_type, position);
- if (container->focus_child != list->undo_focus_child && !list->add_mode)
- {
- gtk_list_unselect_all (list);
- gtk_list_select_child (list, container->focus_child);
- }
- }
- else
- gtk_list_move_focus_child (list, scroll_type, position);
-}
-
-
-/* Private GtkList Scroll/Focus Functions :
- *
- * gtk_list_move_focus_child
- * gtk_list_horizontal_timeout
- * gtk_list_vertical_timeout
- */
-static void
-gtk_list_move_focus_child (GtkList *list,
- GtkScrollType scroll_type,
- gfloat position)
-{
- GtkContainer *container;
- GList *work;
- GtkWidget *item;
- GtkAdjustment *adj;
- gint new_value;
-
- g_return_if_fail (list != 0);
- g_return_if_fail (GTK_IS_LIST (list));
-
- container = GTK_CONTAINER (list);
-
- if (container->focus_child)
- work = g_list_find (list->children, container->focus_child);
- else
- work = list->children;
-
- if (!work)
- return;
-
- switch (scroll_type)
- {
- case GTK_SCROLL_STEP_BACKWARD:
- work = work->prev;
- if (work)
- gtk_widget_grab_focus (GTK_WIDGET (work->data));
- break;
- case GTK_SCROLL_STEP_FORWARD:
- work = work->next;
- if (work)
- gtk_widget_grab_focus (GTK_WIDGET (work->data));
- break;
- case GTK_SCROLL_PAGE_BACKWARD:
- if (!work->prev)
- return;
-
- item = work->data;
- adj = gtk_object_get_data_by_id (GTK_OBJECT (list), vadjustment_key_id);
-
- if (adj)
- {
- gboolean correct = FALSE;
-
- new_value = adj->value;
-
- if (item->allocation.y <= adj->value)
- {
- new_value = MAX (item->allocation.y + item->allocation.height
- - adj->page_size, adj->lower);
- correct = TRUE;
- }
-
- if (item->allocation.y > new_value)
- for (; work; work = work->prev)
- {
- item = GTK_WIDGET (work->data);
- if (item->allocation.y <= new_value &&
- item->allocation.y + item->allocation.height > new_value)
- break;
- }
- else
- for (; work; work = work->next)
- {
- item = GTK_WIDGET (work->data);
- if (item->allocation.y <= new_value &&
- item->allocation.y + item->allocation.height > new_value)
- break;
- }
-
- if (correct && work && work->next && item->allocation.y < new_value)
- item = work->next->data;
- }
- else
- item = list->children->data;
-
- gtk_widget_grab_focus (item);
- break;
- case GTK_SCROLL_PAGE_FORWARD:
- if (!work->next)
- return;
-
- item = work->data;
- adj = gtk_object_get_data_by_id (GTK_OBJECT (list), vadjustment_key_id);
-
- if (adj)
- {
- gboolean correct = FALSE;
-
- new_value = adj->value;
-
- if (item->allocation.y + item->allocation.height >=
- adj->value + adj->page_size)
- {
- new_value = item->allocation.y;
- correct = TRUE;
- }
-
- new_value = MIN (new_value + adj->page_size, adj->upper);
-
- if (item->allocation.y > new_value)
- for (; work; work = work->prev)
- {
- item = GTK_WIDGET (work->data);
- if (item->allocation.y <= new_value &&
- item->allocation.y + item->allocation.height > new_value)
- break;
- }
- else
- for (; work; work = work->next)
- {
- item = GTK_WIDGET (work->data);
- if (item->allocation.y <= new_value &&
- item->allocation.y + item->allocation.height > new_value)
- break;
- }
-
- if (correct && work && work->prev &&
- item->allocation.y + item->allocation.height - 1 > new_value)
- item = work->prev->data;
- }
- else
- item = g_list_last (work)->data;
-
- gtk_widget_grab_focus (item);
- break;
- case GTK_SCROLL_JUMP:
- new_value = GTK_WIDGET(list)->allocation.height * CLAMP (position, 0, 1);
-
- for (item = NULL, work = list->children; work; work =work->next)
- {
- item = GTK_WIDGET (work->data);
- if (item->allocation.y <= new_value &&
- item->allocation.y + item->allocation.height > new_value)
- break;
- }
-
- gtk_widget_grab_focus (item);
- break;
- default:
- break;
- }
-}
-
-static void
-do_fake_motion (GtkWidget *list)
-{
- GdkEvent *event = gdk_event_new (GDK_MOTION_NOTIFY);
-
- event->motion.send_event = TRUE;
-
- gtk_list_motion_notify (list, (GdkEventMotion *)event);
- gdk_event_free (event);
-}
-
-static gint
-gtk_list_horizontal_timeout (GtkWidget *list)
-{
- GTK_LIST (list)->htimer = 0;
- do_fake_motion (list);
-
- return FALSE;
-}
-
-static gint
-gtk_list_vertical_timeout (GtkWidget *list)
-{
- GTK_LIST (list)->vtimer = 0;
- do_fake_motion (list);
-
- return FALSE;
-}
-
-
-/* Private GtkListItem Signal Functions :
- *
- * gtk_list_signal_toggle_focus_row
- * gtk_list_signal_select_all
- * gtk_list_signal_unselect_all
- * gtk_list_signal_undo_selection
- * gtk_list_signal_start_selection
- * gtk_list_signal_end_selection
- * gtk_list_signal_extend_selection
- * gtk_list_signal_scroll_horizontal
- * gtk_list_signal_scroll_vertical
- * gtk_list_signal_toggle_add_mode
- * gtk_list_signal_item_select
- * gtk_list_signal_item_deselect
- * gtk_list_signal_item_toggle
- */
-static void
-gtk_list_signal_toggle_focus_row (GtkListItem *list_item,
- GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_toggle_focus_row (list);
-}
-
-static void
-gtk_list_signal_select_all (GtkListItem *list_item,
- GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_select_all (list);
-}
-
-static void
-gtk_list_signal_unselect_all (GtkListItem *list_item,
- GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_unselect_all (list);
-}
-
-static void
-gtk_list_signal_undo_selection (GtkListItem *list_item,
- GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_undo_selection (list);
-}
-
-static void
-gtk_list_signal_start_selection (GtkListItem *list_item,
- GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_start_selection (list);
-}
-
-static void
-gtk_list_signal_end_selection (GtkListItem *list_item,
- GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_end_selection (list);
-}
-
-static void
-gtk_list_signal_extend_selection (GtkListItem *list_item,
- GtkScrollType scroll_type,
- gfloat position,
- gboolean auto_start_selection,
- GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_extend_selection (list, scroll_type, position,
- auto_start_selection);
-}
-
-static void
-gtk_list_signal_scroll_horizontal (GtkListItem *list_item,
- GtkScrollType scroll_type,
- gfloat position,
- GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_scroll_horizontal (list, scroll_type, position);
-}
-
-static void
-gtk_list_signal_scroll_vertical (GtkListItem *list_item,
- GtkScrollType scroll_type,
- gfloat position,
- GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_scroll_vertical (list, scroll_type, position);
-}
-
-static void
-gtk_list_signal_toggle_add_mode (GtkListItem *list_item,
- GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_toggle_add_mode (list);
-}
-
-static void
-gtk_list_signal_item_select (GtkListItem *list_item,
- GtkList *list)
-{
- GList *selection;
- GList *tmp_list;
- GList *sel_list;
-
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (GTK_WIDGET (list_item)->state != GTK_STATE_SELECTED)
- return;
-
- switch (list->selection_mode)
- {
- case GTK_SELECTION_SINGLE:
- case GTK_SELECTION_BROWSE:
- sel_list = NULL;
- selection = list->selection;
-
- while (selection)
- {
- tmp_list = selection;
- selection = selection->next;
-
- if (tmp_list->data == list_item)
- sel_list = tmp_list;
- else
- gtk_list_item_deselect (GTK_LIST_ITEM (tmp_list->data));
- }
-
- if (!sel_list)
- {
- list->selection = g_list_prepend (list->selection, list_item);
- g_object_ref (list_item);
- }
- gtk_signal_emit (GTK_OBJECT (list), list_signals[SELECTION_CHANGED]);
- break;
- case GTK_SELECTION_MULTIPLE:
- if (list->anchor >= 0)
- return;
- }
-}
-
-static void
-gtk_list_signal_item_deselect (GtkListItem *list_item,
- GtkList *list)
-{
- GList *node;
-
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- if (GTK_WIDGET (list_item)->state != GTK_STATE_NORMAL)
- return;
-
- node = g_list_find (list->selection, list_item);
-
- if (node)
- {
- list->selection = g_list_remove_link (list->selection, node);
- g_list_free_1 (node);
- g_object_unref (list_item);
- gtk_signal_emit (GTK_OBJECT (list), list_signals[SELECTION_CHANGED]);
- }
-}
-
-static void
-gtk_list_signal_item_toggle (GtkListItem *list_item,
- GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST_ITEM (list_item));
- g_return_if_fail (GTK_IS_LIST (list));
-
- if ((list->selection_mode == GTK_SELECTION_BROWSE ||
- list->selection_mode == GTK_SELECTION_MULTIPLE) &&
- GTK_WIDGET (list_item)->state == GTK_STATE_NORMAL)
- {
- gtk_widget_set_state (GTK_WIDGET (list_item), GTK_STATE_SELECTED);
- return;
- }
-
- switch (GTK_WIDGET (list_item)->state)
- {
- case GTK_STATE_SELECTED:
- gtk_list_signal_item_select (list_item, list);
- break;
- case GTK_STATE_NORMAL:
- gtk_list_signal_item_deselect (list_item, list);
- break;
- default:
- break;
- }
-}
-
-static void
-gtk_list_signal_drag_begin (GtkWidget *widget,
- GdkDragContext *context,
- GtkList *list)
-{
- g_return_if_fail (GTK_IS_LIST_ITEM (widget));
- g_return_if_fail (GTK_IS_LIST (list));
-
- gtk_list_drag_begin (GTK_WIDGET (list), context);
-}
-
-static void
-gtk_list_drag_begin (GtkWidget *widget,
- GdkDragContext *context)
-{
- GtkList *list;
-
- g_return_if_fail (GTK_IS_LIST (widget));
- g_return_if_fail (context != NULL);
-
- list = GTK_LIST (widget);
-
- if (list->drag_selection)
- {
- gtk_list_end_drag_selection (list);
-
- switch (list->selection_mode)
- {
- case GTK_SELECTION_MULTIPLE:
- gtk_list_end_selection (list);
- break;
- case GTK_SELECTION_SINGLE:
- list->undo_focus_child = NULL;
- break;
- default:
- break;
- }
- }
-}
-
-#include "gtkaliasdef.c"