diff options
Diffstat (limited to 'widgets/table/e-table-header-item.c')
-rw-r--r-- | widgets/table/e-table-header-item.c | 2227 |
1 files changed, 0 insertions, 2227 deletions
diff --git a/widgets/table/e-table-header-item.c b/widgets/table/e-table-header-item.c deleted file mode 100644 index 44d3979e1c..0000000000 --- a/widgets/table/e-table-header-item.c +++ /dev/null @@ -1,2227 +0,0 @@ -/* - * This program 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) version 3. - * - * This program is distributed in the hope that it will be useful, - * but WITHOUT ANY WARRANTY; without even the implied warranty of - * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU - * Lesser General Public License for more details. - * - * You should have received a copy of the GNU Lesser General Public - * License along with the program; if not, see <http://www.gnu.org/licenses/> - * - * - * Authors: - * Chris Lahey <clahey@ximian.com> - * Miguel de Icaza <miguel@gnu.org> - * - * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) - * - */ - -#ifdef HAVE_CONFIG_H -#include <config.h> -#endif - -#include <string.h> - -#include <gtk/gtk.h> -#include <libgnomecanvas/libgnomecanvas.h> -#include <gdk-pixbuf/gdk-pixbuf.h> -#include <gdk/gdkkeysyms.h> - -#include <glib/gi18n.h> -#include "e-util/e-util.h" -#include "libevolution-utils/e-xml-utils.h" -#include "misc/e-canvas.h" - -#include "e-popup-menu.h" -#include "e-table.h" -#include "e-table-col-dnd.h" -#include "e-table-config.h" -#include "e-table-defines.h" -#include "e-table-field-chooser-dialog.h" -#include "e-table-header.h" -#include "e-table-header-utils.h" - -#include "e-table-header-item.h" - -#include "arrow-up.xpm" -#include "arrow-down.xpm" - -enum { - BUTTON_PRESSED, - LAST_SIGNAL -}; - -static guint ethi_signals[LAST_SIGNAL] = { 0, }; - -#define ARROW_DOWN_HEIGHT 16 -#define ARROW_PTR 7 - -/* Defines the tolerance for proximity of the column division to the cursor position */ -#define TOLERANCE 4 - -#define ETHI_RESIZING(x) ((x)->resize_col != -1) - -#define ethi_get_type e_table_header_item_get_type -G_DEFINE_TYPE (ETableHeaderItem, ethi, GNOME_TYPE_CANVAS_ITEM) - -#define d(x) - -static void ethi_drop_table_header (ETableHeaderItem *ethi); - -/* - * They display the arrows for the drop location. - */ - -static GtkWidget *arrow_up, *arrow_down; - -enum { - PROP_0, - PROP_TABLE_HEADER, - PROP_FULL_HEADER, - PROP_DND_CODE, - PROP_TABLE_FONT_DESC, - PROP_SORT_INFO, - PROP_TABLE, - PROP_TREE -}; - -enum { - ET_SCROLL_UP = 1 << 0, - ET_SCROLL_DOWN = 1 << 1, - ET_SCROLL_LEFT = 1 << 2, - ET_SCROLL_RIGHT = 1 << 3 -}; - -static void scroll_off (ETableHeaderItem *ethi); -static void scroll_on (ETableHeaderItem *ethi, guint scroll_direction); - -static void -ethi_dispose (GObject *object) -{ - ETableHeaderItem *ethi = E_TABLE_HEADER_ITEM (object); - - ethi_drop_table_header (ethi); - - scroll_off (ethi); - - if (ethi->resize_cursor) { - g_object_unref (ethi->resize_cursor); - ethi->resize_cursor = NULL; - } - - if (ethi->dnd_code) { - g_free (ethi->dnd_code); - ethi->dnd_code = NULL; - } - - if (ethi->sort_info) { - if (ethi->sort_info_changed_id) - g_signal_handler_disconnect ( - ethi->sort_info, ethi->sort_info_changed_id); - if (ethi->group_info_changed_id) - g_signal_handler_disconnect ( - ethi->sort_info, ethi->group_info_changed_id); - g_object_unref (ethi->sort_info); - ethi->sort_info = NULL; - } - - if (ethi->full_header) - g_object_unref (ethi->full_header); - ethi->full_header = NULL; - - if (ethi->etfcd.widget) - g_object_remove_weak_pointer ( - G_OBJECT (ethi->etfcd.widget), ði->etfcd.pointer); - - if (ethi->config) - g_object_unref (ethi->config); - ethi->config = NULL; - - /* Chain up to parent's dispose() method. */ - G_OBJECT_CLASS (ethi_parent_class)->dispose (object); -} - -static gint -e_table_header_item_get_height (ETableHeaderItem *ethi) -{ - ETableHeader *eth; - gint numcols, col; - gint maxheight; - - g_return_val_if_fail (ethi != NULL, 0); - g_return_val_if_fail (E_IS_TABLE_HEADER_ITEM (ethi), 0); - - eth = ethi->eth; - numcols = e_table_header_count (eth); - - maxheight = 0; - - for (col = 0; col < numcols; col++) { - ETableCol *ecol = e_table_header_get_column (eth, col); - gint height; - - height = e_table_header_compute_height ( - ecol, GTK_WIDGET (GNOME_CANVAS_ITEM (ethi)->canvas)); - - if (height > maxheight) - maxheight = height; - } - - return maxheight; -} - -static void -ethi_update (GnomeCanvasItem *item, - const cairo_matrix_t *i2c, - gint flags) -{ - ETableHeaderItem *ethi = E_TABLE_HEADER_ITEM (item); - gdouble x1, y1, x2, y2; - - if (GNOME_CANVAS_ITEM_CLASS (ethi_parent_class)->update) - GNOME_CANVAS_ITEM_CLASS (ethi_parent_class)->update ( - item, i2c, flags); - - if (ethi->sort_info) - ethi->group_indent_width = - e_table_sort_info_grouping_get_count (ethi->sort_info) - * GROUP_INDENT; - else - ethi->group_indent_width = 0; - - ethi->width = - e_table_header_total_width (ethi->eth) + - ethi->group_indent_width; - - x1 = y1 = 0; - x2 = ethi->width; - y2 = ethi->height; - - gnome_canvas_matrix_transform_rect (i2c, &x1, &y1, &x2, &y2); - - if (item->x1 != x1 || - item->y1 != y1 || - item->x2 != x2 || - item->y2 != y2) { - gnome_canvas_request_redraw ( - item->canvas, - item->x1, item->y1, - item->x2, item->y2); - item->x1 = x1; - item->y1 = y1; - item->x2 = x2; - item->y2 = y2; - } - gnome_canvas_request_redraw ( - item->canvas, item->x1, item->y1, item->x2, item->y2); -} - -static void -ethi_font_set (ETableHeaderItem *ethi, - PangoFontDescription *font_desc) -{ - if (ethi->font_desc) - pango_font_description_free (ethi->font_desc); - - ethi->font_desc = pango_font_description_copy (font_desc); - - ethi->height = e_table_header_item_get_height (ethi); - e_canvas_item_request_reflow (GNOME_CANVAS_ITEM (ethi)); -} - -static void -ethi_drop_table_header (ETableHeaderItem *ethi) -{ - GObject *header; - - if (!ethi->eth) - return; - - header = G_OBJECT (ethi->eth); - g_signal_handler_disconnect (header, ethi->structure_change_id); - g_signal_handler_disconnect (header, ethi->dimension_change_id); - - g_object_unref (header); - ethi->eth = NULL; - ethi->width = 0; -} - -static void -structure_changed (ETableHeader *header, - ETableHeaderItem *ethi) -{ - gnome_canvas_item_request_update (GNOME_CANVAS_ITEM (ethi)); -} - -static void -dimension_changed (ETableHeader *header, - gint col, - ETableHeaderItem *ethi) -{ - gnome_canvas_item_request_update (GNOME_CANVAS_ITEM (ethi)); -} - -static void -ethi_add_table_header (ETableHeaderItem *ethi, - ETableHeader *header) -{ - ethi->eth = header; - g_object_ref (ethi->eth); - - ethi->height = e_table_header_item_get_height (ethi); - - ethi->structure_change_id = g_signal_connect ( - header, "structure_change", - G_CALLBACK (structure_changed), ethi); - ethi->dimension_change_id = g_signal_connect ( - header, "dimension_change", - G_CALLBACK (dimension_changed), ethi); - e_canvas_item_request_reflow (GNOME_CANVAS_ITEM (ethi)); - gnome_canvas_item_request_update (GNOME_CANVAS_ITEM (ethi)); -} - -static void -ethi_sort_info_changed (ETableSortInfo *sort_info, - ETableHeaderItem *ethi) -{ - gnome_canvas_item_request_update (GNOME_CANVAS_ITEM (ethi)); -} - -static void -ethi_set_property (GObject *object, - guint property_id, - const GValue *value, - GParamSpec *pspec) -{ - GnomeCanvasItem *item; - ETableHeaderItem *ethi; - - item = GNOME_CANVAS_ITEM (object); - ethi = E_TABLE_HEADER_ITEM (object); - - switch (property_id) { - case PROP_TABLE_HEADER: - ethi_drop_table_header (ethi); - ethi_add_table_header (ethi, E_TABLE_HEADER (g_value_get_object (value))); - break; - - case PROP_FULL_HEADER: - if (ethi->full_header) - g_object_unref (ethi->full_header); - ethi->full_header = E_TABLE_HEADER (g_value_get_object (value)); - if (ethi->full_header) - g_object_ref (ethi->full_header); - break; - - case PROP_DND_CODE: - g_free (ethi->dnd_code); - ethi->dnd_code = g_strdup (g_value_get_string (value)); - break; - - case PROP_TABLE_FONT_DESC: - ethi_font_set (ethi, g_value_get_boxed (value)); - break; - - case PROP_SORT_INFO: - if (ethi->sort_info) { - if (ethi->sort_info_changed_id) - g_signal_handler_disconnect ( - ethi->sort_info, - ethi->sort_info_changed_id); - - if (ethi->group_info_changed_id) - g_signal_handler_disconnect ( - ethi->sort_info, - ethi->group_info_changed_id); - g_object_unref (ethi->sort_info); - } - ethi->sort_info = g_value_get_object (value); - g_object_ref (ethi->sort_info); - ethi->sort_info_changed_id = - g_signal_connect ( - ethi->sort_info, "sort_info_changed", - G_CALLBACK (ethi_sort_info_changed), ethi); - ethi->group_info_changed_id = - g_signal_connect ( - ethi->sort_info, "group_info_changed", - G_CALLBACK (ethi_sort_info_changed), ethi); - break; - case PROP_TABLE: - if (g_value_get_object (value)) - ethi->table = E_TABLE (g_value_get_object (value)); - else - ethi->table = NULL; - break; - case PROP_TREE: - if (g_value_get_object (value)) - ethi->tree = E_TREE (g_value_get_object (value)); - else - ethi->tree = NULL; - break; - } - gnome_canvas_item_request_update (item); -} - -static void -ethi_get_property (GObject *object, - guint property_id, - GValue *value, - GParamSpec *pspec) -{ - ETableHeaderItem *ethi; - - ethi = E_TABLE_HEADER_ITEM (object); - - switch (property_id) { - case PROP_FULL_HEADER: - g_value_set_object (value, ethi->full_header); - break; - case PROP_DND_CODE: - g_value_set_string (value, ethi->dnd_code); - break; - default: - G_OBJECT_WARN_INVALID_PROPERTY_ID (object, property_id, pspec); - break; - } -} - -static gint -ethi_find_col_by_x (ETableHeaderItem *ethi, - gint x) -{ - const gint cols = e_table_header_count (ethi->eth); - gint x1 = 0; - gint col; - - d (g_print ("%s:%d: x = %d, x1 = %d\n", __FUNCTION__, __LINE__, x, x1)); - - x1 += ethi->group_indent_width; - - if (x < x1) { - d (g_print ("%s:%d: Returning 0\n", __FUNCTION__, __LINE__)); - return 0; - } - - for (col = 0; col < cols; col++) { - ETableCol *ecol = e_table_header_get_column (ethi->eth, col); - - if ((x >= x1) && (x <= x1 + ecol->width)) { - d (g_print ("%s:%d: Returning %d\n", __FUNCTION__, __LINE__, col)); - return col; - } - - x1 += ecol->width; - } - d (g_print ("%s:%d: Returning %d\n", __FUNCTION__, __LINE__, cols - 1)); - return cols - 1; -} - -static gint -ethi_find_col_by_x_nearest (ETableHeaderItem *ethi, - gint x) -{ - const gint cols = e_table_header_count (ethi->eth); - gint x1 = 0; - gint col; - - x1 += ethi->group_indent_width; - - if (x < x1) - return 0; - - for (col = 0; col < cols; col++) { - ETableCol *ecol = e_table_header_get_column (ethi->eth, col); - - x1 += (ecol->width / 2); - - if (x <= x1) - return col; - - x1 += (ecol->width + 1) / 2; - } - return col; -} - -static void -ethi_remove_drop_marker (ETableHeaderItem *ethi) -{ - if (ethi->drag_mark == -1) - return; - - gtk_widget_hide (arrow_up); - gtk_widget_hide (arrow_down); - - ethi->drag_mark = -1; -} - -static GtkWidget * -make_shaped_window_from_xpm (const gchar **xpm) -{ - GdkPixbuf *pixbuf; - GtkWidget *win, *pix; - - pixbuf = gdk_pixbuf_new_from_xpm_data (xpm); - - win = gtk_window_new (GTK_WINDOW_POPUP); - gtk_window_set_type_hint (GTK_WINDOW (win), GDK_WINDOW_TYPE_HINT_NOTIFICATION); - - pix = gtk_image_new_from_pixbuf (pixbuf); - gtk_widget_realize (win); - gtk_container_add (GTK_CONTAINER (win), pix); - - g_object_unref (pixbuf); - - return win; -} - -static void -ethi_add_drop_marker (ETableHeaderItem *ethi, - gint col, - gboolean recreate) -{ - GnomeCanvas *canvas; - GtkAdjustment *adjustment; - GdkWindow *window; - gint rx, ry; - gint x; - - if (!recreate && ethi->drag_mark == col) - return; - - ethi->drag_mark = col; - - x = e_table_header_col_diff (ethi->eth, 0, col); - if (col > 0) - x += ethi->group_indent_width; - - if (!arrow_up) { - arrow_up = make_shaped_window_from_xpm (arrow_up_xpm); - arrow_down = make_shaped_window_from_xpm (arrow_down_xpm); - } - - canvas = GNOME_CANVAS_ITEM (ethi)->canvas; - window = gtk_widget_get_window (GTK_WIDGET (canvas)); - gdk_window_get_origin (window, &rx, &ry); - - adjustment = gtk_scrollable_get_hadjustment (GTK_SCROLLABLE (canvas)); - rx -= gtk_adjustment_get_value (adjustment); - - adjustment = gtk_scrollable_get_vadjustment (GTK_SCROLLABLE (canvas)); - ry -= gtk_adjustment_get_value (adjustment); - - gtk_window_move ( - GTK_WINDOW (arrow_down), - rx + x - ARROW_PTR, - ry - ARROW_DOWN_HEIGHT); - gtk_widget_show_all (arrow_down); - - gtk_window_move ( - GTK_WINDOW (arrow_up), - rx + x - ARROW_PTR, - ry + ethi->height); - gtk_widget_show_all (arrow_up); -} - -static void -ethi_add_destroy_marker (ETableHeaderItem *ethi) -{ - gdouble x1; - - if (ethi->remove_item) - g_object_run_dispose (G_OBJECT (ethi->remove_item)); - - x1 = (gdouble) e_table_header_col_diff (ethi->eth, 0, ethi->drag_col); - if (ethi->drag_col > 0) - x1 += ethi->group_indent_width; - - ethi->remove_item = gnome_canvas_item_new ( - GNOME_CANVAS_GROUP (GNOME_CANVAS_ITEM (ethi)->canvas->root), - gnome_canvas_rect_get_type (), - "x1", x1 + 1, - "y1", (gdouble) 1, - "x2", (gdouble) x1 + e_table_header_col_diff ( - ethi->eth, ethi->drag_col, ethi->drag_col + 1) - 2, - - "y2", (gdouble) ethi->height - 2, - "fill_color_rgba", 0xFF000080, - NULL); -} - -static void -ethi_remove_destroy_marker (ETableHeaderItem *ethi) -{ - if (!ethi->remove_item) - return; - - g_object_run_dispose (G_OBJECT (ethi->remove_item)); - ethi->remove_item = NULL; -} - -#if 0 -static gboolean -moved (ETableHeaderItem *ethi, - guint col, - guint model_col) -{ - if (col == -1) - return TRUE; - ecol = e_table_header_get_column (ethi->eth, col); - if (ecol->col_idx == model_col) - return FALSE; - if (col > 0) { - ecol = e_table_header_get_column (ethi->eth, col - 1); - if (ecol->col_idx == model_col) - return FALSE; - } - return TRUE; -} -#endif - -static void -do_drag_motion (ETableHeaderItem *ethi, - GdkDragContext *context, - gint x, - gint y, - guint time, - gboolean recreate) -{ - if ((x >= 0) && (x <= (ethi->width)) && - (y >= 0) && (y <= (ethi->height))) { - GdkDragAction suggested_action; - gint col; - d (g_print ("In header\n")); - - col = ethi_find_col_by_x_nearest (ethi, x); - suggested_action = gdk_drag_context_get_suggested_action (context); - - if (ethi->drag_col != -1 && (col == ethi->drag_col || - col == ethi->drag_col + 1)) { - if (ethi->drag_col != -1) - ethi_remove_destroy_marker (ethi); - - ethi_remove_drop_marker (ethi); - gdk_drag_status (context, suggested_action, time); - } - else if (col != -1) { - if (ethi->drag_col != -1) - ethi_remove_destroy_marker (ethi); - - ethi_add_drop_marker (ethi, col, recreate); - gdk_drag_status (context, suggested_action, time); - } else { - ethi_remove_drop_marker (ethi); - if (ethi->drag_col != -1) - ethi_add_destroy_marker (ethi); - } - } else { - ethi_remove_drop_marker (ethi); - if (ethi->drag_col != -1) - ethi_add_destroy_marker (ethi); - } -} - -static gboolean -scroll_timeout (gpointer data) -{ - ETableHeaderItem *ethi = data; - gint dx = 0; - GtkAdjustment *adjustment; - GtkScrollable *scrollable; - gdouble hadjustment_value; - gdouble vadjustment_value; - gdouble page_size; - gdouble lower; - gdouble upper; - gdouble value; - - if (ethi->scroll_direction & ET_SCROLL_RIGHT) - dx += 20; - if (ethi->scroll_direction & ET_SCROLL_LEFT) - dx -= 20; - - scrollable = GTK_SCROLLABLE (GNOME_CANVAS_ITEM (ethi)->canvas); - - adjustment = gtk_scrollable_get_hadjustment (scrollable); - hadjustment_value = gtk_adjustment_get_value (adjustment); - - adjustment = gtk_scrollable_get_vadjustment (scrollable); - vadjustment_value = gtk_adjustment_get_value (adjustment); - - value = hadjustment_value; - - adjustment = gtk_scrollable_get_hadjustment (scrollable); - page_size = gtk_adjustment_get_page_size (adjustment); - lower = gtk_adjustment_get_lower (adjustment); - upper = gtk_adjustment_get_upper (adjustment); - - gtk_adjustment_set_value ( - adjustment, CLAMP ( - hadjustment_value + dx, lower, upper - page_size)); - - hadjustment_value = gtk_adjustment_get_value (adjustment); - - if (hadjustment_value != value) - do_drag_motion ( - ethi, - ethi->last_drop_context, - ethi->last_drop_x + hadjustment_value, - ethi->last_drop_y + vadjustment_value, - ethi->last_drop_time, - TRUE); - - return TRUE; -} - -static void -scroll_on (ETableHeaderItem *ethi, - guint scroll_direction) -{ - if (ethi->scroll_idle_id == 0 || scroll_direction != ethi->scroll_direction) { - if (ethi->scroll_idle_id != 0) - g_source_remove (ethi->scroll_idle_id); - ethi->scroll_direction = scroll_direction; - ethi->scroll_idle_id = g_timeout_add (100, scroll_timeout, ethi); - } -} - -static void -scroll_off (ETableHeaderItem *ethi) -{ - if (ethi->scroll_idle_id) { - g_source_remove (ethi->scroll_idle_id); - ethi->scroll_idle_id = 0; - } -} - -static void -context_destroyed (gpointer data) -{ - ETableHeaderItem *ethi = data; - - ethi->last_drop_x = 0; - ethi->last_drop_y = 0; - ethi->last_drop_time = 0; - ethi->last_drop_context = NULL; - scroll_off (ethi); - - g_object_unref (ethi); -} - -static void -context_connect (ETableHeaderItem *ethi, - GdkDragContext *context) -{ - if (g_dataset_get_data (context, "e-table-header-item") == NULL) - g_dataset_set_data_full ( - context, "e-table-header-item", - g_object_ref (ethi), context_destroyed); -} - -static gboolean -ethi_drag_motion (GtkWidget *widget, - GdkDragContext *context, - gint x, - gint y, - guint time, - ETableHeaderItem *ethi) -{ - GtkAllocation allocation; - GtkAdjustment *adjustment; - GList *targets; - gdouble hadjustment_value; - gdouble vadjustment_value; - gchar *droptype, *headertype; - guint direction = 0; - - gdk_drag_status (context, 0, time); - - targets = gdk_drag_context_list_targets (context); - droptype = gdk_atom_name (GDK_POINTER_TO_ATOM (targets->data)); - headertype = g_strdup_printf ( - "%s-%s", TARGET_ETABLE_COL_TYPE, ethi->dnd_code); - - if (strcmp (droptype, headertype) != 0) { - g_free (headertype); - return FALSE; - } - - g_free (headertype); - - gtk_widget_get_allocation (widget, &allocation); - - if (x < 20) - direction |= ET_SCROLL_LEFT; - if (x > allocation.width - 20) - direction |= ET_SCROLL_RIGHT; - - ethi->last_drop_x = x; - ethi->last_drop_y = y; - ethi->last_drop_time = time; - ethi->last_drop_context = context; - context_connect (ethi, context); - - adjustment = gtk_scrollable_get_hadjustment (GTK_SCROLLABLE (widget)); - hadjustment_value = gtk_adjustment_get_value (adjustment); - - adjustment = gtk_scrollable_get_vadjustment (GTK_SCROLLABLE (widget)); - vadjustment_value = gtk_adjustment_get_value (adjustment); - - do_drag_motion ( - ethi, context, - x + hadjustment_value, - y + vadjustment_value, - time, FALSE); - - if (direction != 0) - scroll_on (ethi, direction); - else - scroll_off (ethi); - - return TRUE; -} - -static void -ethi_drag_end (GtkWidget *canvas, - GdkDragContext *context, - ETableHeaderItem *ethi) -{ - ethi_remove_drop_marker (ethi); - ethi_remove_destroy_marker (ethi); - ethi->drag_col = -1; - scroll_off (ethi); -} - -static void -ethi_drag_data_received (GtkWidget *canvas, - GdkDragContext *drag_context, - gint x, - gint y, - GtkSelectionData *selection_data, - guint info, - guint time, - ETableHeaderItem *ethi) -{ - const guchar *data; - gint found = FALSE; - gint count; - gint column; - gint drop_col; - gint i; - - data = gtk_selection_data_get_data (selection_data); - - if (data != NULL) { - count = e_table_header_count (ethi->eth); - column = atoi ((gchar *) data); - drop_col = ethi->drop_col; - ethi->drop_col = -1; - - if (column >= 0) { - for (i = 0; i < count; i++) { - ETableCol *ecol = e_table_header_get_column (ethi->eth, i); - if (ecol->col_idx == column) { - e_table_header_move (ethi->eth, i, drop_col); - found = TRUE; - break; - } - } - if (!found) { - count = e_table_header_count (ethi->full_header); - for (i = 0; i < count; i++) { - ETableCol *ecol; - - ecol = e_table_header_get_column ( - ethi->full_header, i); - - if (ecol->col_idx == column) { - e_table_header_add_column ( - ethi->eth, ecol, - drop_col); - break; - } - } - } - } - } - ethi_remove_drop_marker (ethi); - gnome_canvas_item_request_update (GNOME_CANVAS_ITEM (ethi)); -} - -static void -ethi_drag_data_get (GtkWidget *canvas, - GdkDragContext *context, - GtkSelectionData *selection_data, - guint info, - guint time, - ETableHeaderItem *ethi) -{ - if (ethi->drag_col != -1) { - ETableCol *ecol = e_table_header_get_column (ethi->eth, ethi->drag_col); - - gchar *string = g_strdup_printf ("%d", ecol->col_idx); - gtk_selection_data_set ( - selection_data, - GDK_SELECTION_TYPE_STRING, - sizeof (string[0]), - (guchar *) string, - strlen (string)); - g_free (string); - } -} - -static gboolean -ethi_drag_drop (GtkWidget *canvas, - GdkDragContext *context, - gint x, - gint y, - guint time, - ETableHeaderItem *ethi) -{ - gboolean successful = FALSE; - - if ((x >= 0) && (x <= (ethi->width)) && - (y >= 0) && (y <= (ethi->height))) { - gint col; - - col = ethi_find_col_by_x_nearest (ethi, x); - - ethi_add_drop_marker (ethi, col, FALSE); - - ethi->drop_col = col; - - if (col != -1) { - gchar *target = g_strdup_printf ( - "%s-%s", TARGET_ETABLE_COL_TYPE, ethi->dnd_code); - d (g_print ("ethi - %s\n", target)); - gtk_drag_get_data ( - canvas, context, - gdk_atom_intern (target, FALSE), - time); - g_free (target); - } - } - gtk_drag_finish (context, successful, successful, time); - scroll_off (ethi); - return successful; -} - -static void -ethi_drag_leave (GtkWidget *widget, - GdkDragContext *context, - guint time, - ETableHeaderItem *ethi) -{ - ethi_remove_drop_marker (ethi); - if (ethi->drag_col != -1) - ethi_add_destroy_marker (ethi); -} - -static void -ethi_realize (GnomeCanvasItem *item) -{ - ETableHeaderItem *ethi = E_TABLE_HEADER_ITEM (item); - GtkStyle *style; - GtkTargetEntry ethi_drop_types[] = { - { (gchar *) TARGET_ETABLE_COL_TYPE, 0, TARGET_ETABLE_COL_HEADER }, - }; - - if (GNOME_CANVAS_ITEM_CLASS (ethi_parent_class)-> realize) - (*GNOME_CANVAS_ITEM_CLASS (ethi_parent_class)->realize)(item); - - style = gtk_widget_get_style (GTK_WIDGET (item->canvas)); - - if (!ethi->font_desc) - ethi_font_set (ethi, style->font_desc); - - /* - * Now, configure DnD - */ - ethi_drop_types[0].target = g_strdup_printf ( - "%s-%s", ethi_drop_types[0].target, ethi->dnd_code); - gtk_drag_dest_set ( - GTK_WIDGET (item->canvas), 0, ethi_drop_types, - G_N_ELEMENTS (ethi_drop_types), GDK_ACTION_MOVE); - g_free ((gpointer) ethi_drop_types[0].target); - - /* Drop signals */ - ethi->drag_motion_id = g_signal_connect ( - item->canvas, "drag_motion", - G_CALLBACK (ethi_drag_motion), ethi); - ethi->drag_leave_id = g_signal_connect ( - item->canvas, "drag_leave", - G_CALLBACK (ethi_drag_leave), ethi); - ethi->drag_drop_id = g_signal_connect ( - item->canvas, "drag_drop", - G_CALLBACK (ethi_drag_drop), ethi); - ethi->drag_data_received_id = g_signal_connect ( - item->canvas, "drag_data_received", - G_CALLBACK (ethi_drag_data_received), ethi); - - /* Drag signals */ - ethi->drag_end_id = g_signal_connect ( - item->canvas, "drag_end", - G_CALLBACK (ethi_drag_end), ethi); - ethi->drag_data_get_id = g_signal_connect ( - item->canvas, "drag_data_get", - G_CALLBACK (ethi_drag_data_get), ethi); - -} - -static void -ethi_unrealize (GnomeCanvasItem *item) -{ - ETableHeaderItem *ethi = E_TABLE_HEADER_ITEM (item); - - if (ethi->font_desc != NULL) { - pango_font_description_free (ethi->font_desc); - ethi->font_desc = NULL; - } - - g_signal_handler_disconnect (item->canvas, ethi->drag_motion_id); - g_signal_handler_disconnect (item->canvas, ethi->drag_leave_id); - g_signal_handler_disconnect (item->canvas, ethi->drag_drop_id); - g_signal_handler_disconnect (item->canvas, ethi->drag_data_received_id); - - g_signal_handler_disconnect (item->canvas, ethi->drag_end_id); - g_signal_handler_disconnect (item->canvas, ethi->drag_data_get_id); - - gtk_drag_dest_unset (GTK_WIDGET (item->canvas)); - - if (GNOME_CANVAS_ITEM_CLASS (ethi_parent_class)->unrealize) - (*GNOME_CANVAS_ITEM_CLASS (ethi_parent_class)->unrealize)(item); -} - -static void -ethi_draw (GnomeCanvasItem *item, - cairo_t *cr, - gint x, - gint y, - gint width, - gint height) -{ - ETableHeaderItem *ethi = E_TABLE_HEADER_ITEM (item); - GnomeCanvas *canvas = item->canvas; - const gint cols = e_table_header_count (ethi->eth); - gint x1, x2; - gint col; - GHashTable *arrows = g_hash_table_new (NULL, NULL); - GtkStyleContext *context; - - if (ethi->sort_info) { - gint length; - gint i; - - length = e_table_sort_info_grouping_get_count (ethi->sort_info); - for (i = 0; i < length; i++) { - ETableSortColumn column; - - column = e_table_sort_info_grouping_get_nth ( - ethi->sort_info, i); - - g_hash_table_insert ( - arrows, - GINT_TO_POINTER ((gint) column.column), - GINT_TO_POINTER ( - column.ascending ? - E_TABLE_COL_ARROW_DOWN : - E_TABLE_COL_ARROW_UP)); - } - - length = e_table_sort_info_sorting_get_count (ethi->sort_info); - for (i = 0; i < length; i++) { - ETableSortColumn column; - - column = e_table_sort_info_sorting_get_nth ( - ethi->sort_info, i); - - g_hash_table_insert ( - arrows, - GINT_TO_POINTER ((gint) column.column), - GINT_TO_POINTER ( - column.ascending ? - E_TABLE_COL_ARROW_DOWN : - E_TABLE_COL_ARROW_UP)); - } - } - - ethi->width = e_table_header_total_width (ethi->eth) + ethi->group_indent_width; - x1 = x2 = 0; - x2 += ethi->group_indent_width; - - context = gtk_widget_get_style_context (GTK_WIDGET (canvas)); - - for (col = 0; col < cols; col++, x1 = x2) { - ETableCol *ecol = e_table_header_get_column (ethi->eth, col); - gint col_width; - GtkRegionFlags flags = 0; - - col_width = ecol->width; - - x2 += col_width; - - if (x1 > (x + width)) - break; - - if (x2 < x) - continue; - - if (x2 <= x1) - continue; - - if (((col + 1) % 2) == 0) - flags |= GTK_REGION_EVEN; - else - flags |= GTK_REGION_ODD; - - if (col == 0) - flags |= GTK_REGION_FIRST; - - if (col + 1 == cols) - flags |= GTK_REGION_LAST; - - gtk_style_context_save (context); - gtk_style_context_add_region ( - context, GTK_STYLE_REGION_COLUMN_HEADER, flags); - - e_table_header_draw_button ( - cr, ecol, GTK_WIDGET (canvas), - x1 - x, -y, width, height, - x2 - x1, ethi->height, - (ETableColArrow) g_hash_table_lookup ( - arrows, GINT_TO_POINTER (ecol->col_idx))); - - gtk_style_context_restore (context); - } - - g_hash_table_destroy (arrows); -} - -static GnomeCanvasItem * -ethi_point (GnomeCanvasItem *item, - gdouble x, - gdouble y, - gint cx, - gint cy) -{ - return item; -} - -/* - * is_pointer_on_division: - * - * Returns whether @pos is a column header division; If @the_total is not NULL, - * then the actual position is returned here. If @return_ecol is not NULL, - * then the ETableCol that actually contains this point is returned here - */ -static gboolean -is_pointer_on_division (ETableHeaderItem *ethi, - gint pos, - gint *the_total, - gint *return_col) -{ - const gint cols = e_table_header_count (ethi->eth); - gint col, total; - - total = 0; - for (col = 0; col < cols; col++) { - ETableCol *ecol = e_table_header_get_column (ethi->eth, col); - - if (col == 0) - total += ethi->group_indent_width; - - total += ecol->width; - - if ((total - TOLERANCE < pos) && (pos < total + TOLERANCE)) { - if (return_col) - *return_col = col; - if (the_total) - *the_total = total; - - return TRUE; - } - if (return_col) - *return_col = col; - - if (total > pos + TOLERANCE) - return FALSE; - } - - return FALSE; -} - -#define convert(c,sx,sy,x,y) gnome_canvas_w2c (c,sx,sy,x,y) - -static void -set_cursor (ETableHeaderItem *ethi, - gint pos) -{ - GnomeCanvas *canvas; - GdkWindow *window; - gboolean resizable = FALSE; - gint col; - - canvas = GNOME_CANVAS_ITEM (ethi)->canvas; - window = gtk_widget_get_window (GTK_WIDGET (canvas)); - - /* We might be invoked before we are realized */ - if (window == NULL) - return; - - if (is_pointer_on_division (ethi, pos, NULL, &col)) { - gint last_col = ethi->eth->col_count - 1; - ETableCol *ecol = e_table_header_get_column (ethi->eth, col); - - /* Last column is not resizable */ - if (ecol->resizable && col != last_col) { - gint c = col + 1; - - /* Column is not resizable if all columns after it - * are also not resizable */ - for (; c <= last_col; c++) { - ETableCol *ecol2; - - ecol2 = e_table_header_get_column (ethi->eth, c); - if (ecol2->resizable) { - resizable = TRUE; - break; - } - } - } - } - - if (resizable) - gdk_window_set_cursor (window, ethi->resize_cursor); - else - gdk_window_set_cursor (window, NULL); -} - -static void -ethi_end_resize (ETableHeaderItem *ethi) -{ - ethi->resize_col = -1; - ethi->resize_guide = GINT_TO_POINTER (0); - - if (ethi->table) - e_table_thaw_state_change (ethi->table); - else if (ethi->tree) - e_tree_thaw_state_change (ethi->tree); - - gnome_canvas_item_request_update (GNOME_CANVAS_ITEM (ethi)); -} - -static gboolean -ethi_maybe_start_drag (ETableHeaderItem *ethi, - GdkEventMotion *event) -{ - if (!ethi->maybe_drag) - return FALSE; - - if (ethi->eth->col_count < 2) { - ethi->maybe_drag = FALSE; - return FALSE; - } - - if (MAX (abs (ethi->click_x - event->x), - abs (ethi->click_y - event->y)) <= 3) - return FALSE; - - return TRUE; -} - -static void -ethi_start_drag (ETableHeaderItem *ethi, - GdkEvent *event) -{ - GtkWidget *widget = GTK_WIDGET (GNOME_CANVAS_ITEM (ethi)->canvas); - GtkTargetList *list; - GdkDragContext *context; - ETableCol *ecol; - gint col_width; - cairo_surface_t *s; - cairo_t *cr; - - gint group_indent = 0; - GHashTable *arrows = g_hash_table_new (NULL, NULL); - - GtkTargetEntry ethi_drag_types[] = { - { (gchar *) TARGET_ETABLE_COL_TYPE, 0, TARGET_ETABLE_COL_HEADER }, - }; - - widget = GTK_WIDGET (GNOME_CANVAS_ITEM (ethi)->canvas); - ethi->drag_col = ethi_find_col_by_x (ethi, event->motion.x); - - if (ethi->drag_col == -1) - return; - - if (ethi->sort_info) { - gint length = e_table_sort_info_grouping_get_count (ethi->sort_info); - gint i; - for (i = 0; i < length; i++) { - ETableSortColumn column = - e_table_sort_info_grouping_get_nth ( - ethi->sort_info, i); - group_indent++; - g_hash_table_insert ( - arrows, - GINT_TO_POINTER ((gint) column.column), - GINT_TO_POINTER ( - column.ascending ? - E_TABLE_COL_ARROW_DOWN : - E_TABLE_COL_ARROW_UP)); - } - length = e_table_sort_info_sorting_get_count (ethi->sort_info); - for (i = 0; i < length; i++) { - ETableSortColumn column = - e_table_sort_info_sorting_get_nth ( - ethi->sort_info, i); - - g_hash_table_insert ( - arrows, - GINT_TO_POINTER ((gint) column.column), - GINT_TO_POINTER ( - column.ascending ? - E_TABLE_COL_ARROW_DOWN : - E_TABLE_COL_ARROW_UP)); - } - } - - ethi_drag_types[0].target = g_strdup_printf ( - "%s-%s", ethi_drag_types[0].target, ethi->dnd_code); - list = gtk_target_list_new ( - ethi_drag_types, G_N_ELEMENTS (ethi_drag_types)); - context = gtk_drag_begin (widget, list, GDK_ACTION_MOVE, 1, event); - g_free ((gpointer) ethi_drag_types[0].target); - - ecol = e_table_header_get_column (ethi->eth, ethi->drag_col); - col_width = ecol->width; - s = cairo_image_surface_create (CAIRO_FORMAT_ARGB32, col_width, ethi->height); - cr = cairo_create (s); - - e_table_header_draw_button ( - cr, ecol, - widget, 0, 0, - col_width, ethi->height, - col_width, ethi->height, - (ETableColArrow) g_hash_table_lookup ( - arrows, GINT_TO_POINTER (ecol->col_idx))); - gtk_drag_set_icon_surface (context, s); - cairo_surface_destroy (s); - - ethi->maybe_drag = FALSE; - g_hash_table_destroy (arrows); -} - -typedef struct { - ETableHeaderItem *ethi; - gint col; -} EthiHeaderInfo; - -static void -ethi_popup_sort_ascending (GtkWidget *widget, - EthiHeaderInfo *info) -{ - ETableCol *col; - gint model_col = -1; - gint length; - gint i; - gint found = FALSE; - ETableHeaderItem *ethi = info->ethi; - - col = e_table_header_get_column (ethi->eth, info->col); - if (col->sortable) - model_col = col->col_idx; - - length = e_table_sort_info_grouping_get_count (ethi->sort_info); - for (i = 0; i < length; i++) { - ETableSortColumn column = e_table_sort_info_grouping_get_nth ( - ethi->sort_info, i); - - if (model_col == column.column) { - column.ascending = 1; - e_table_sort_info_grouping_set_nth ( - ethi->sort_info, i, column); - found = 1; - break; - } - } - if (!found) { - length = e_table_sort_info_sorting_get_count ( - ethi->sort_info); - for (i = 0; i < length; i++) { - ETableSortColumn column = - e_table_sort_info_sorting_get_nth ( - ethi->sort_info, i); - if (model_col == column.column || model_col == -1) { - column.ascending = 1; - e_table_sort_info_sorting_set_nth ( - ethi->sort_info, i, column); - found = 1; - if (model_col != -1) - break; - } - } - } - if (!found) { - ETableSortColumn column; - column.column = model_col; - column.ascending = 1; - length = e_table_sort_info_sorting_get_count (ethi->sort_info); - if (length == 0) - length++; - e_table_sort_info_sorting_set_nth (ethi->sort_info, length - 1, column); - } -} - -static void -ethi_popup_sort_descending (GtkWidget *widget, - EthiHeaderInfo *info) -{ - ETableCol *col; - gint model_col=-1; - gint length; - gint i; - gint found = FALSE; - ETableHeaderItem *ethi = info->ethi; - - col = e_table_header_get_column (ethi->eth, info->col); - if (col->sortable) - model_col = col->col_idx; - - length = e_table_sort_info_grouping_get_count (ethi->sort_info); - for (i = 0; i < length; i++) { - ETableSortColumn column = e_table_sort_info_grouping_get_nth ( - ethi->sort_info, i); - if (model_col == column.column) { - column.ascending = 0; - e_table_sort_info_grouping_set_nth ( - ethi->sort_info, i, column); - found = 1; - break; - } - } - if (!found) { - length = e_table_sort_info_sorting_get_count (ethi->sort_info); - for (i = 0; i < length; i++) { - ETableSortColumn column = - e_table_sort_info_sorting_get_nth ( - ethi->sort_info, i); - - if (model_col == column.column || model_col == -1) { - column.ascending = 0; - e_table_sort_info_sorting_set_nth ( - ethi->sort_info, i, column); - found = 1; - if (model_col != -1) - break; - } - } - } - if (!found) { - ETableSortColumn column; - column.column = model_col; - column.ascending = 0; - length = e_table_sort_info_sorting_get_count (ethi->sort_info); - if (length == 0) - length++; - e_table_sort_info_sorting_set_nth ( - ethi->sort_info, length - 1, column); - } -} - -static void -ethi_popup_unsort (GtkWidget *widget, - EthiHeaderInfo *info) -{ - ETableHeaderItem *ethi = info->ethi; - - e_table_sort_info_grouping_truncate (ethi->sort_info, 0); - e_table_sort_info_sorting_truncate (ethi->sort_info, 0); -} - -static void -ethi_popup_group_field (GtkWidget *widget, - EthiHeaderInfo *info) -{ - ETableCol *col; - gint model_col; - ETableHeaderItem *ethi = info->ethi; - ETableSortColumn column; - - col = e_table_header_get_column (ethi->eth, info->col); - model_col = col->col_idx; - - column.column = model_col; - column.ascending = 1; - e_table_sort_info_grouping_set_nth (ethi->sort_info, 0, column); - e_table_sort_info_grouping_truncate (ethi->sort_info, 1); -} - -static void -ethi_popup_group_box (GtkWidget *widget, - EthiHeaderInfo *info) -{ -} - -static void -ethi_popup_remove_column (GtkWidget *widget, - EthiHeaderInfo *info) -{ - e_table_header_remove (info->ethi->eth, info->col); -} - -static void -ethi_popup_field_chooser (GtkWidget *widget, - EthiHeaderInfo *info) -{ - GtkWidget *etfcd = info->ethi->etfcd.widget; - - if (etfcd) { - gtk_window_present (GTK_WINDOW (etfcd)); - - return; - } - - info->ethi->etfcd.widget = e_table_field_chooser_dialog_new (); - etfcd = info->ethi->etfcd.widget; - - g_object_add_weak_pointer (G_OBJECT (etfcd), &info->ethi->etfcd.pointer); - - g_object_set ( - info->ethi->etfcd.widget, - "full_header", info->ethi->full_header, - "header", info->ethi->eth, - "dnd_code", info->ethi->dnd_code, - NULL); - - gtk_widget_show (etfcd); -} - -static void -ethi_popup_alignment (GtkWidget *widget, - EthiHeaderInfo *info) -{ -} - -static void -ethi_popup_best_fit (GtkWidget *widget, - EthiHeaderInfo *info) -{ - ETableHeaderItem *ethi = info->ethi; - gint width; - - g_signal_emit_by_name ( - ethi->eth, - "request_width", - info->col, &width); - /* Add 10 to stop it from "..."ing */ - e_table_header_set_size (ethi->eth, info->col, width + 10); - - gnome_canvas_item_request_update (GNOME_CANVAS_ITEM (ethi)); - -} - -static void -ethi_popup_format_columns (GtkWidget *widget, - EthiHeaderInfo *info) -{ -} - -static void -config_destroyed (gpointer data, - GObject *where_object_was) -{ - ETableHeaderItem *ethi = data; - ethi->config = NULL; -} - -static void -apply_changes (ETableConfig *config, - ETableHeaderItem *ethi) -{ - gchar *state = e_table_state_save_to_string (config->state); - - if (ethi->table) - e_table_set_state (ethi->table, state); - if (ethi->tree) - e_tree_set_state (ethi->tree, state); - g_free (state); - - gtk_dialog_set_response_sensitive ( - GTK_DIALOG (config->dialog_toplevel), - GTK_RESPONSE_APPLY, FALSE); -} - -static void -ethi_popup_customize_view (GtkWidget *widget, - EthiHeaderInfo *info) -{ - ETableHeaderItem *ethi = info->ethi; - ETableState *state; - ETableSpecification *spec; - - if (ethi->config) - e_table_config_raise (E_TABLE_CONFIG (ethi->config)); - else { - if (ethi->table) { - state = e_table_get_state_object (ethi->table); - spec = ethi->table->spec; - } else if (ethi->tree) { - state = e_tree_get_state_object (ethi->tree); - spec = e_tree_get_spec (ethi->tree); - } else - return; - - ethi->config = e_table_config_new ( - _("Customize Current View"), - spec, state, GTK_WINDOW (gtk_widget_get_toplevel (widget))); - g_object_weak_ref ( - G_OBJECT (ethi->config), - config_destroyed, ethi); - g_signal_connect ( - ethi->config, "changed", - G_CALLBACK (apply_changes), ethi); - } -} - -static void -free_popup_info (GtkWidget *w, - EthiHeaderInfo *info) -{ - g_free (info); -} - -/* Bit 1 is always disabled. */ -/* Bit 2 is disabled if not "sortable". */ -/* Bit 4 is disabled if we don't have a pointer to our table object. */ -static EPopupMenu ethi_context_menu[] = { - E_POPUP_ITEM ( - N_("Sort _Ascending"), - G_CALLBACK (ethi_popup_sort_ascending), 2), - E_POPUP_ITEM ( - N_("Sort _Descending"), - G_CALLBACK (ethi_popup_sort_descending), 2), - E_POPUP_ITEM ( - N_("_Unsort"), G_CALLBACK (ethi_popup_unsort), 0), - E_POPUP_SEPARATOR, - E_POPUP_ITEM ( - N_("Group By This _Field"), - G_CALLBACK (ethi_popup_group_field), 16), - E_POPUP_ITEM ( - N_("Group By _Box"), - G_CALLBACK (ethi_popup_group_box), 128), - E_POPUP_SEPARATOR, - E_POPUP_ITEM ( - N_("Remove This _Column"), - G_CALLBACK (ethi_popup_remove_column), 8), - E_POPUP_ITEM ( - N_("Add a C_olumn..."), - G_CALLBACK (ethi_popup_field_chooser), 0), - E_POPUP_SEPARATOR, - E_POPUP_ITEM ( - N_("A_lignment"), - G_CALLBACK (ethi_popup_alignment), 128), - E_POPUP_ITEM ( - N_("B_est Fit"), - G_CALLBACK (ethi_popup_best_fit), 2), - E_POPUP_ITEM ( - N_("Format Column_s..."), - G_CALLBACK (ethi_popup_format_columns), 128), - E_POPUP_SEPARATOR, - E_POPUP_ITEM ( - N_("Custo_mize Current View..."), - G_CALLBACK (ethi_popup_customize_view), 4), - E_POPUP_TERMINATOR -}; - -static void -sort_by_id (GtkWidget *menu_item, - ETableHeaderItem *ethi) -{ - ETableCol *ecol; - gboolean clearfirst; - gint col; - - col = GPOINTER_TO_INT (g_object_get_data ( - G_OBJECT (menu_item), "col-number")); - ecol = e_table_header_get_column (ethi->full_header, col); - clearfirst = e_table_sort_info_sorting_get_count (ethi->sort_info) > 1; - - if (!clearfirst && ecol && - e_table_sort_info_sorting_get_count (ethi->sort_info) == 1) { - ETableSortColumn column; - - column = e_table_sort_info_sorting_get_nth (ethi->sort_info, 0); - clearfirst = ecol->sortable && ecol->col_idx != column.column; - } - - if (clearfirst) - e_table_sort_info_sorting_truncate (ethi->sort_info, 0); - - ethi_change_sort_state (ethi, ecol); -} - -static void -popup_custom (GtkWidget *menu_item, - EthiHeaderInfo *info) -{ - ethi_popup_customize_view (menu_item, info); -} - -static void -ethi_header_context_menu (ETableHeaderItem *ethi, - GdkEvent *button_event) -{ - EthiHeaderInfo *info = g_new (EthiHeaderInfo, 1); - GtkMenu *popup; - gint ncol, sort_count, sort_col; - GtkWidget *menu_item, *sub_menu; - ETableSortColumn column; - gboolean ascending = TRUE; - gdouble event_x_win = 0; - gdouble event_y_win = 0; - guint event_button = 0; - guint32 event_time; - - d (g_print ("ethi_header_context_menu: \n")); - - gdk_event_get_button (button_event, &event_button); - gdk_event_get_coords (button_event, &event_x_win, &event_y_win); - event_time = gdk_event_get_time (button_event); - - info->ethi = ethi; - info->col = ethi_find_col_by_x (ethi, event_x_win); - - popup = e_popup_menu_create_with_domain ( - ethi_context_menu, - 1 + - ((ethi->table || ethi->tree) ? 0 : 4) + - ((e_table_header_count (ethi->eth) > 1) ? 0 : 8), - ((e_table_sort_info_get_can_group (ethi->sort_info)) ? 0 : 16) + - 128, info, GETTEXT_PACKAGE); - - menu_item = gtk_menu_item_new_with_mnemonic (_("_Sort By")); - gtk_widget_show (menu_item); - sub_menu = gtk_menu_new (); - gtk_widget_show (sub_menu); - gtk_menu_item_set_submenu (GTK_MENU_ITEM (menu_item), sub_menu); - gtk_menu_shell_prepend (GTK_MENU_SHELL (popup), menu_item); - - sort_count = e_table_sort_info_sorting_get_count (ethi->sort_info); - - if (sort_count > 1 || sort_count < 1) - sort_col = -1; /* Custom sorting */ - else { - column = e_table_sort_info_sorting_get_nth (ethi->sort_info, 0); - sort_col = column.column; - ascending = column.ascending; - } - - /* Custom */ - menu_item = gtk_check_menu_item_new_with_mnemonic (_("_Custom")); - gtk_widget_show (menu_item); - gtk_menu_shell_prepend (GTK_MENU_SHELL (sub_menu), menu_item); - if (sort_col == -1) - gtk_check_menu_item_set_active (GTK_CHECK_MENU_ITEM (menu_item), TRUE); - gtk_check_menu_item_set_draw_as_radio (GTK_CHECK_MENU_ITEM (menu_item), TRUE); - g_signal_connect ( - menu_item, "activate", - G_CALLBACK (popup_custom), info); - - /* Show a seperator */ - menu_item = gtk_separator_menu_item_new (); - gtk_widget_show (menu_item); - gtk_menu_shell_prepend (GTK_MENU_SHELL (sub_menu), menu_item); - /* Headers */ - for (ncol = 0; ncol < ethi->full_header->col_count; ncol++) - { - gchar *text = NULL; - - if (!ethi->full_header->columns[ncol]->sortable || - ethi->full_header->columns[ncol]->disabled) - continue; - - if (ncol == sort_col) { - text = g_strdup_printf ( - "%s (%s)", - ethi->full_header->columns[ncol]->text, - ascending ? _("Ascending"):_("Descending")); - menu_item = gtk_check_menu_item_new_with_label (text); - g_free (text); - } else - menu_item = gtk_check_menu_item_new_with_label ( - ethi->full_header->columns[ncol]->text); - - gtk_widget_show (menu_item); - gtk_menu_shell_prepend (GTK_MENU_SHELL (sub_menu), menu_item); - - if (ncol == sort_col) - gtk_check_menu_item_set_active ( - GTK_CHECK_MENU_ITEM (menu_item), TRUE); - gtk_check_menu_item_set_draw_as_radio ( - GTK_CHECK_MENU_ITEM (menu_item), TRUE); - g_object_set_data ( - G_OBJECT (menu_item), "col-number", - GINT_TO_POINTER (ncol)); - g_signal_connect ( - menu_item, "activate", - G_CALLBACK (sort_by_id), ethi); - } - - g_object_ref_sink (popup); - g_signal_connect ( - popup, "selection-done", - G_CALLBACK (free_popup_info), info); - - gtk_menu_popup ( - GTK_MENU (popup), - NULL, NULL, NULL, NULL, - event_button, event_time); -} - -static void -ethi_button_pressed (ETableHeaderItem *ethi, - GdkEvent *button_event) -{ - g_signal_emit (ethi, ethi_signals[BUTTON_PRESSED], 0, button_event); -} - -void -ethi_change_sort_state (ETableHeaderItem *ethi, - ETableCol *col) -{ - gint model_col = -1; - gint length; - gint i; - gboolean found = FALSE; - - if (col == NULL) - return; - - if (col->sortable) - model_col = col->col_idx; - - length = e_table_sort_info_grouping_get_count (ethi->sort_info); - for (i = 0; i < length; i++) { - ETableSortColumn column; - - column = e_table_sort_info_grouping_get_nth ( - ethi->sort_info, i); - - if (model_col == column.column || model_col == -1) { - gint ascending = column.ascending; - ascending = !ascending; - column.ascending = ascending; - e_table_sort_info_grouping_set_nth (ethi->sort_info, i, column); - found = TRUE; - if (model_col != -1) - break; - } - } - - if (!found) { - length = e_table_sort_info_sorting_get_count (ethi->sort_info); - for (i = 0; i < length; i++) { - ETableSortColumn column; - - column = e_table_sort_info_sorting_get_nth ( - ethi->sort_info, i); - - if (model_col == column.column || model_col == -1) { - gint ascending = column.ascending; - - if (ascending == 0 && model_col != -1) { - /* - * This means the user has clicked twice - * already, lets kill sorting of this column now. - */ - gint j; - - for (j = i + 1; j < length; j++) - e_table_sort_info_sorting_set_nth ( - ethi->sort_info, j - 1, - e_table_sort_info_sorting_get_nth ( - ethi->sort_info, j)); - - e_table_sort_info_sorting_truncate ( - ethi->sort_info, length - 1); - length--; - i--; - } else { - ascending = !ascending; - column.ascending = ascending; - e_table_sort_info_sorting_set_nth ( - ethi->sort_info, i, column); - } - found = TRUE; - if (model_col != -1) - break; - } - } - } - - if (!found && model_col != -1) { - ETableSortColumn column; - column.column = model_col; - column.ascending = 1; - e_table_sort_info_sorting_truncate (ethi->sort_info, 0); - e_table_sort_info_sorting_set_nth (ethi->sort_info, 0, column); - } -} - -/* - * Handles the events on the ETableHeaderItem, particularly it handles resizing - */ -static gint -ethi_event (GnomeCanvasItem *item, - GdkEvent *event) -{ - ETableHeaderItem *ethi = E_TABLE_HEADER_ITEM (item); - GnomeCanvas *canvas = item->canvas; - GdkWindow *window; - const gboolean resizing = ETHI_RESIZING (ethi); - gint x, y, start, col; - gint was_maybe_drag = 0; - GdkModifierType event_state = 0; - guint event_button = 0; - guint event_keyval = 0; - gdouble event_x_win = 0; - gdouble event_y_win = 0; - guint32 event_time; - - /* Don't fetch the device here. GnomeCanvas frequently emits - * synthesized events, and calling gdk_event_get_device() on them - * will trigger a runtime warning. Fetch the device where needed. */ - gdk_event_get_button (event, &event_button); - gdk_event_get_coords (event, &event_x_win, &event_y_win); - gdk_event_get_keyval (event, &event_keyval); - gdk_event_get_state (event, &event_state); - event_time = gdk_event_get_time (event); - - switch (event->type) { - case GDK_ENTER_NOTIFY: - convert (canvas, event_x_win, event_y_win, &x, &y); - set_cursor (ethi, x); - break; - - case GDK_LEAVE_NOTIFY: - window = gtk_widget_get_window (GTK_WIDGET (canvas)); - gdk_window_set_cursor (window, NULL); - break; - - case GDK_MOTION_NOTIFY: - - convert (canvas, event_x_win, event_y_win, &x, &y); - if (resizing) { - gint new_width; - - if (ethi->resize_guide == NULL) { - GdkDevice *event_device; - - /* Quick hack until I actually bind the views */ - ethi->resize_guide = GINT_TO_POINTER (1); - - event_device = gdk_event_get_device (event); - - gnome_canvas_item_grab ( - item, - GDK_POINTER_MOTION_MASK | - GDK_BUTTON_RELEASE_MASK, - ethi->resize_cursor, - event_device, - event_time); - } - - new_width = x - ethi->resize_start_pos; - - e_table_header_set_size (ethi->eth, ethi->resize_col, new_width); - - gnome_canvas_item_request_update (GNOME_CANVAS_ITEM (ethi)); - } else if (ethi_maybe_start_drag (ethi, &event->motion)) { - ethi_start_drag (ethi, event); - } else - set_cursor (ethi, x); - break; - - case GDK_BUTTON_PRESS: - if (event_button > 3) - return FALSE; - - convert (canvas, event_x_win, event_y_win, &x, &y); - - if (is_pointer_on_division (ethi, x, &start, &col) && - event_button == 1) { - ETableCol *ecol; - - /* - * Record the important bits. - * - * By setting resize_pos to a non -1 value, - * we know that we are being resized (used in the - * other event handlers). - */ - ecol = e_table_header_get_column (ethi->eth, col); - - if (!ecol->resizable) - break; - ethi->resize_col = col; - ethi->resize_start_pos = start - ecol->width; - ethi->resize_min_width = ecol->min_width; - - if (ethi->table) - e_table_freeze_state_change (ethi->table); - else if (ethi->tree) - e_tree_freeze_state_change (ethi->tree); - } else { - if (event_button == 1) { - ethi->click_x = event_x_win; - ethi->click_y = event_y_win; - ethi->maybe_drag = TRUE; - is_pointer_on_division (ethi, x, &start, &col); - ethi->selected_col = col; - if (gtk_widget_get_can_focus (GTK_WIDGET (item->canvas))) - e_canvas_item_grab_focus (item, TRUE); - } else if (event_button == 3) { - ethi_header_context_menu (ethi, event); - } else - ethi_button_pressed (ethi, event); - } - break; - - case GDK_2BUTTON_PRESS: - if (!resizing) - break; - - if (event_button != 1) - break; - else { - gint width = 0; - g_signal_emit_by_name ( - ethi->eth, - "request_width", - (gint) ethi->resize_col, &width); - /* Add 10 to stop it from "..."ing */ - e_table_header_set_size (ethi->eth, ethi->resize_col, width + 10); - - gnome_canvas_item_request_update (GNOME_CANVAS_ITEM (ethi)); - ethi->maybe_drag = FALSE; - } - break; - - case GDK_BUTTON_RELEASE: { - gboolean needs_ungrab = FALSE; - - was_maybe_drag = ethi->maybe_drag; - - ethi->maybe_drag = FALSE; - - if (ethi->resize_col != -1) { - needs_ungrab = (ethi->resize_guide != NULL); - ethi_end_resize (ethi); - } else if (was_maybe_drag && ethi->sort_info) { - ETableCol *ecol; - - col = ethi_find_col_by_x (ethi, event_x_win); - ecol = e_table_header_get_column (ethi->eth, col); - ethi_change_sort_state (ethi, ecol); - } - - if (needs_ungrab) - gnome_canvas_item_ungrab (item, event_time); - - break; - } - case GDK_KEY_PRESS: - if ((event_keyval == GDK_KEY_F10) && (event_state & GDK_SHIFT_MASK)) { - EthiHeaderInfo *info = g_new (EthiHeaderInfo, 1); - ETableCol *ecol; - GtkMenu *popup; - - info->ethi = ethi; - info->col = ethi->selected_col; - ecol = e_table_header_get_column (ethi->eth, info->col); - - popup = e_popup_menu_create_with_domain ( - ethi_context_menu, - 1 + - (ecol->sortable ? 0 : 2) + - ((ethi->table || ethi->tree) ? 0 : 4) + - ((e_table_header_count (ethi->eth) > 1) ? 0 : 8), - ((e_table_sort_info_get_can_group ( - ethi->sort_info)) ? 0 : 16) + - 128, info, GETTEXT_PACKAGE); - g_object_ref_sink (popup); - g_signal_connect ( - popup, "selection-done", - G_CALLBACK (free_popup_info), info); - gtk_menu_popup ( - GTK_MENU (popup), - NULL, NULL, NULL, NULL, - 0, GDK_CURRENT_TIME); - } else if (event_keyval == GDK_KEY_space) { - ETableCol *ecol; - - ecol = e_table_header_get_column (ethi->eth, ethi->selected_col); - ethi_change_sort_state (ethi, ecol); - } else if ((event_keyval == GDK_KEY_Right) || - (event_keyval == GDK_KEY_KP_Right)) { - ETableCol *ecol; - - if ((ethi->selected_col < 0) || - (ethi->selected_col >= ethi->eth->col_count - 1)) - ethi->selected_col = 0; - else - ethi->selected_col++; - ecol = e_table_header_get_column (ethi->eth, ethi->selected_col); - ethi_change_sort_state (ethi, ecol); - } else if ((event_keyval == GDK_KEY_Left) || - (event_keyval == GDK_KEY_KP_Left)) { - ETableCol *ecol; - - if ((ethi->selected_col <= 0) || - (ethi->selected_col >= ethi->eth->col_count)) - ethi->selected_col = ethi->eth->col_count - 1; - else - ethi->selected_col--; - ecol = e_table_header_get_column (ethi->eth, ethi->selected_col); - ethi_change_sort_state (ethi, ecol); - } - break; - - default: - return FALSE; - } - return TRUE; -} - -static void -ethi_class_init (ETableHeaderItemClass *class) -{ - GnomeCanvasItemClass *item_class = GNOME_CANVAS_ITEM_CLASS (class); - GObjectClass *object_class = G_OBJECT_CLASS (class); - - object_class->dispose = ethi_dispose; - object_class->set_property = ethi_set_property; - object_class->get_property = ethi_get_property; - - item_class->update = ethi_update; - item_class->realize = ethi_realize; - item_class->unrealize = ethi_unrealize; - item_class->draw = ethi_draw; - item_class->point = ethi_point; - item_class->event = ethi_event; - - g_object_class_install_property ( - object_class, - PROP_DND_CODE, - g_param_spec_string ( - "dnd_code", - "DnD code", - NULL, - NULL, - G_PARAM_READWRITE)); - - g_object_class_install_property ( - object_class, - PROP_TABLE_FONT_DESC, - g_param_spec_boxed ( - "font-desc", - "Font Description", - NULL, - PANGO_TYPE_FONT_DESCRIPTION, - G_PARAM_WRITABLE)); - - g_object_class_install_property ( - object_class, - PROP_FULL_HEADER, - g_param_spec_object ( - "full_header", - "Full Header", - NULL, - E_TYPE_TABLE_HEADER, - G_PARAM_READWRITE)); - - g_object_class_install_property ( - object_class, - PROP_TABLE_HEADER, - g_param_spec_object ( - "ETableHeader", - "Header", - NULL, - E_TYPE_TABLE_HEADER, - G_PARAM_WRITABLE)); - - g_object_class_install_property ( - object_class, - PROP_SORT_INFO, - g_param_spec_object ( - "sort_info", - "Sort Info", - NULL, - E_TYPE_TABLE_SORT_INFO, - G_PARAM_WRITABLE)); - - g_object_class_install_property ( - object_class, - PROP_TABLE, - g_param_spec_object ( - "table", - "Table", - NULL, - E_TYPE_TABLE, - G_PARAM_WRITABLE)); - - g_object_class_install_property ( - object_class, - PROP_TREE, - g_param_spec_object ( - "tree", - "Tree", - NULL, - E_TYPE_TREE, - G_PARAM_WRITABLE)); - - ethi_signals[BUTTON_PRESSED] = g_signal_new ( - "button_pressed", - G_OBJECT_CLASS_TYPE (object_class), - G_SIGNAL_RUN_LAST, - G_STRUCT_OFFSET (ETableHeaderItemClass, button_pressed), - NULL, NULL, - g_cclosure_marshal_VOID__BOXED, - G_TYPE_NONE, 1, - GDK_TYPE_EVENT | G_SIGNAL_TYPE_STATIC_SCOPE); -} - -static void -ethi_init (ETableHeaderItem *ethi) -{ - GnomeCanvasItem *item = GNOME_CANVAS_ITEM (ethi); - - ethi->resize_cursor = gdk_cursor_new (GDK_SB_H_DOUBLE_ARROW); - - ethi->resize_col = -1; - - item->x1 = 0; - item->y1 = 0; - item->x2 = 0; - item->y2 = 0; - - ethi->drag_col = -1; - ethi->drag_mark = -1; - - ethi->sort_info = NULL; - - ethi->sort_info_changed_id = 0; - ethi->group_info_changed_id = 0; - - ethi->group_indent_width = 0; - ethi->table = NULL; - ethi->tree = NULL; - - ethi->selected_col = 0; -} - |