/* * 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 * * * Authors: * Chris Lahey * * Copyright (C) 1999-2008 Novell, Inc. (www.novell.com) * */ #include #include #include #include #include #include #include #include "a11y/e-table/gal-a11y-e-table-click-to-add.h" #include "text/e-text.h" #include #include "e-util/e-util.h" #include "misc/e-canvas-utils.h" #include "misc/e-canvas.h" #include "e-table-click-to-add.h" #include "e-table-defines.h" #include "e-table-header.h" #include "e-table-one.h" enum { CURSOR_CHANGE, STYLE_SET, LAST_SIGNAL }; static guint etcta_signals [LAST_SIGNAL] = { 0 }; /* workaround for avoiding APi breakage */ #define etcta_get_type e_table_click_to_add_get_type G_DEFINE_TYPE (ETableClickToAdd, etcta, GNOME_TYPE_CANVAS_GROUP) #define ELEMENTS(x) (sizeof (x) / sizeof (x[0])) enum { PROP_0, PROP_HEADER, PROP_MODEL, PROP_MESSAGE, PROP_WIDTH, PROP_HEIGHT }; static void etcta_cursor_change (GtkObject *object, gint row, gint col, ETableClickToAdd *etcta) { g_signal_emit (etcta, etcta_signals [CURSOR_CHANGE], 0, row, col); } static void etcta_style_set (ETableClickToAdd *etcta, GtkStyle *previous_style) { GtkWidget *widget = GTK_WIDGET(GNOME_CANVAS_ITEM(etcta)->canvas); if (etcta->rect) { gnome_canvas_item_set (etcta->rect, "outline_color_gdk", &widget->style->fg[GTK_STATE_NORMAL], "fill_color_gdk", &widget->style->bg[GTK_STATE_NORMAL], NULL ); } if (etcta->text) gnome_canvas_item_set (etcta->text, "fill_color_gdk", &widget->style->text[GTK_STATE_NORMAL], NULL); } static void etcta_add_table_header (ETableClickToAdd *etcta, ETableHeader *header) { etcta->eth = header; if (etcta->eth) g_object_ref (etcta->eth); if (etcta->row) gnome_canvas_item_set(GNOME_CANVAS_ITEM(etcta->row), "ETableHeader", header, NULL); } static void etcta_drop_table_header (ETableClickToAdd *etcta) { if (!etcta->eth) return; g_object_unref (etcta->eth); etcta->eth = NULL; } static void etcta_add_one (ETableClickToAdd *etcta, ETableModel *one) { etcta->one = one; if (etcta->one) g_object_ref (etcta->one); if (etcta->row) gnome_canvas_item_set(GNOME_CANVAS_ITEM(etcta->row), "ETableModel", one, NULL); g_object_set(etcta->selection, "model", one, NULL); } static void etcta_drop_one (ETableClickToAdd *etcta) { if (!etcta->one) return; g_object_unref (etcta->one); etcta->one = NULL; g_object_set(etcta->selection, "model", NULL, NULL); } static void etcta_add_model (ETableClickToAdd *etcta, ETableModel *model) { etcta->model = model; if (etcta->model) g_object_ref (etcta->model); } static void etcta_drop_model (ETableClickToAdd *etcta) { etcta_drop_one (etcta); if (!etcta->model) return; g_object_unref (etcta->model); etcta->model = NULL; } static void etcta_add_message (ETableClickToAdd *etcta, const gchar *message) { etcta->message = g_strdup(message); } static void etcta_drop_message (ETableClickToAdd *etcta) { g_free(etcta->message); etcta->message = NULL; } static void etcta_dispose (GObject *object) { ETableClickToAdd *etcta = E_TABLE_CLICK_TO_ADD (object); etcta_drop_table_header (etcta); etcta_drop_model (etcta); etcta_drop_message (etcta); if (etcta->selection) g_object_unref (etcta->selection); etcta->selection = NULL; if (G_OBJECT_CLASS (etcta_parent_class)->dispose) (*G_OBJECT_CLASS (etcta_parent_class)->dispose) (object); } static void etcta_set_property (GObject *object, guint prop_id, const GValue *value, GParamSpec *pspec) { GnomeCanvasItem *item; ETableClickToAdd *etcta; item = GNOME_CANVAS_ITEM (object); etcta = E_TABLE_CLICK_TO_ADD (object); switch (prop_id) { case PROP_HEADER: etcta_drop_table_header (etcta); etcta_add_table_header (etcta, E_TABLE_HEADER(g_value_get_object (value))); break; case PROP_MODEL: etcta_drop_model (etcta); etcta_add_model (etcta, E_TABLE_MODEL(g_value_get_object (value))); break; case PROP_MESSAGE: etcta_drop_message (etcta); etcta_add_message (etcta, g_value_get_string (value)); break; case PROP_WIDTH: etcta->width = g_value_get_double (value); if (etcta->row) gnome_canvas_item_set(etcta->row, "minimum_width", etcta->width, NULL); if (etcta->text) gnome_canvas_item_set(etcta->text, "width", etcta->width - 4, NULL); if (etcta->rect) gnome_canvas_item_set(etcta->rect, "x2", etcta->width - 1, NULL); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); return; } gnome_canvas_item_request_update(item); } static void create_rect_and_text (ETableClickToAdd *etcta) { GtkWidget *widget = GTK_WIDGET (GNOME_CANVAS_ITEM(etcta)->canvas); if (!etcta->rect) etcta->rect = gnome_canvas_item_new(GNOME_CANVAS_GROUP(etcta), gnome_canvas_rect_get_type(), "x1", (double) 0, "y1", (double) 0, "x2", (double) etcta->width - 1, "y2", (double) etcta->height - 1, "outline_color_gdk", &widget->style->fg[GTK_STATE_NORMAL], "fill_color_gdk", &widget->style->bg[GTK_STATE_NORMAL], NULL); if (!etcta->text) etcta->text = gnome_canvas_item_new(GNOME_CANVAS_GROUP(etcta), e_text_get_type(), "text", etcta->message ? etcta->message : "", "anchor", GTK_ANCHOR_NW, "width", etcta->width - 4, "draw_background", FALSE, "fill_color_gdk", &widget->style->text[GTK_STATE_NORMAL], NULL); } static void etcta_get_property (GObject *object, guint prop_id, GValue *value, GParamSpec *pspec) { ETableClickToAdd *etcta; etcta = E_TABLE_CLICK_TO_ADD (object); switch (prop_id) { case PROP_HEADER: g_value_set_object (value, etcta->eth); break; case PROP_MODEL: g_value_set_object (value, etcta->model); break; case PROP_MESSAGE: g_value_set_string (value, g_strdup(etcta->message)); break; case PROP_WIDTH: g_value_set_double (value, etcta->width); break; case PROP_HEIGHT: g_value_set_double (value, etcta->height); break; default: G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec); break; } } static void etcta_realize (GnomeCanvasItem *item) { ETableClickToAdd *etcta = E_TABLE_CLICK_TO_ADD (item); create_rect_and_text (etcta); e_canvas_item_move_absolute (etcta->text, 2, 2); if (GNOME_CANVAS_ITEM_CLASS (etcta_parent_class)->realize) (*GNOME_CANVAS_ITEM_CLASS (etcta_parent_class)->realize)(item); e_canvas_item_request_reflow (item); } static void etcta_unrealize (GnomeCanvasItem *item) { if (GNOME_CANVAS_ITEM_CLASS (etcta_parent_class)->unrealize) (*GNOME_CANVAS_ITEM_CLASS (etcta_parent_class)->unrealize)(item); } static void finish_editing (ETableClickToAdd *etcta); static gint item_key_press (ETableItem *item, gint row, gint col, GdkEvent *event, ETableClickToAdd *etcta) { switch (event->key.keyval) { case GDK_Return: case GDK_KP_Enter: case GDK_ISO_Enter: case GDK_3270_Enter: finish_editing(etcta); return TRUE; } return FALSE; } static void set_initial_selection (ETableClickToAdd *etcta) { e_selection_model_do_something (E_SELECTION_MODEL(etcta->selection), 0, e_table_header_prioritized_column (etcta->eth), 0); } static void finish_editing (ETableClickToAdd *etcta) { if (etcta->row) { ETableModel *one; e_table_item_leave_edit (E_TABLE_ITEM (etcta->row)); e_table_one_commit(E_TABLE_ONE(etcta->one)); etcta_drop_one (etcta); gtk_object_destroy(GTK_OBJECT (etcta->row)); etcta->row = NULL; one = e_table_one_new(etcta->model); etcta_add_one (etcta, one); g_object_unref (one); e_selection_model_clear(E_SELECTION_MODEL(etcta->selection)); etcta->row = gnome_canvas_item_new(GNOME_CANVAS_GROUP(etcta), e_table_item_get_type(), "ETableHeader", etcta->eth, "ETableModel", etcta->one, "minimum_width", etcta->width, "horizontal_draw_grid", TRUE, "vertical_draw_grid", TRUE, "selection_model", etcta->selection, "cursor_mode", E_CURSOR_SPREADSHEET, NULL); g_signal_connect(etcta->row, "key_press", G_CALLBACK(item_key_press), etcta); set_initial_selection (etcta); } } /* * Handles the events on the ETableClickToAdd, particularly it creates the ETableItem and passes in some events. */ static gint etcta_event (GnomeCanvasItem *item, GdkEvent *e) { ETableClickToAdd *etcta = E_TABLE_CLICK_TO_ADD (item); switch (e->type) { case GDK_FOCUS_CHANGE: if (!e->focus_change.in) return TRUE; case GDK_BUTTON_PRESS: if (etcta->text) { gtk_object_destroy(GTK_OBJECT (etcta->text)); etcta->text = NULL; } if (etcta->rect) { gtk_object_destroy(GTK_OBJECT (etcta->rect)); etcta->rect = NULL; } if (!etcta->row) { ETableModel *one; one = e_table_one_new(etcta->model); etcta_add_one (etcta, one); g_object_unref (one); e_selection_model_clear(E_SELECTION_MODEL(etcta->selection)); etcta->row = gnome_canvas_item_new(GNOME_CANVAS_GROUP(item), e_table_item_get_type(), "ETableHeader", etcta->eth, "ETableModel", etcta->one, "minimum_width", etcta->width, "horizontal_draw_grid", TRUE, "vertical_draw_grid", TRUE, "selection_model", etcta->selection, "cursor_mode", E_CURSOR_SPREADSHEET, NULL); g_signal_connect(etcta->row, "key_press", G_CALLBACK (item_key_press), etcta); e_canvas_item_grab_focus (GNOME_CANVAS_ITEM(etcta->row), TRUE); set_initial_selection (etcta); } break; case GDK_KEY_PRESS: switch (e->key.keyval) { case GDK_Tab: case GDK_KP_Tab: case GDK_ISO_Left_Tab: finish_editing (etcta); break; default: return FALSE; case GDK_Escape: if (etcta->row) { e_table_item_leave_edit (E_TABLE_ITEM (etcta->row)); etcta_drop_one (etcta); gtk_object_destroy(GTK_OBJECT (etcta->row)); etcta->row = NULL; create_rect_and_text (etcta); e_canvas_item_move_absolute (etcta->text, 3, 3); } break; } break; default: return FALSE; } return TRUE; } static void etcta_reflow (GnomeCanvasItem *item, gint flags) { ETableClickToAdd *etcta = E_TABLE_CLICK_TO_ADD (item); double old_height = etcta->height; if (etcta->text) { g_object_get(etcta->text, "height", &etcta->height, NULL); etcta->height += 6; } if (etcta->row) { g_object_get(etcta->row, "height", &etcta->height, NULL); } if (etcta->rect) { g_object_set(etcta->rect, "y2", etcta->height - 1, NULL); } if (old_height != etcta->height) e_canvas_item_request_parent_reflow(item); } static void etcta_class_init (ETableClickToAddClass *klass) { GnomeCanvasItemClass *item_class = GNOME_CANVAS_ITEM_CLASS(klass); GObjectClass *object_class = G_OBJECT_CLASS(klass); klass->cursor_change = NULL; klass->style_set = etcta_style_set; object_class->dispose = etcta_dispose; object_class->set_property = etcta_set_property; object_class->get_property = etcta_get_property; item_class->realize = etcta_realize; item_class->unrealize = etcta_unrealize; item_class->event = etcta_event; g_object_class_install_property (object_class, PROP_HEADER, g_param_spec_object ("header", _("Header"), /*_( */"XXX blurb" /*)*/, E_TABLE_HEADER_TYPE, G_PARAM_READWRITE)); g_object_class_install_property (object_class, PROP_MODEL, g_param_spec_object ("model", _("Model"), /*_( */"XXX blurb" /*)*/, E_TABLE_MODEL_TYPE, G_PARAM_READWRITE)); g_object_class_install_property (object_class, PROP_MESSAGE, g_param_spec_string ("message", _("Message"), /*_( */"XXX blurb" /*)*/, NULL, G_PARAM_READWRITE)); g_object_class_install_property (object_class, PROP_WIDTH, g_param_spec_double ("width", _("Width"), /*_( */"XXX blurb" /*)*/, 0.0, G_MAXDOUBLE, 0.0, G_PARAM_READWRITE | G_PARAM_LAX_VALIDATION)); g_object_class_install_property (object_class, PROP_HEIGHT, g_param_spec_double ("height", _("Height"), /*_( */"XXX blurb" /*)*/, 0.0, G_MAXDOUBLE, 0.0, G_PARAM_READABLE | G_PARAM_LAX_VALIDATION)); etcta_signals [CURSOR_CHANGE] = g_signal_new ("cursor_change", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (ETableClickToAddClass, cursor_change), NULL, NULL, e_marshal_VOID__INT_INT, G_TYPE_NONE, 2, G_TYPE_INT, G_TYPE_INT); etcta_signals [STYLE_SET] = g_signal_new ("style_set", G_OBJECT_CLASS_TYPE (object_class), G_SIGNAL_RUN_LAST, G_STRUCT_OFFSET (ETableClickToAddClass, style_set), NULL, NULL, g_cclosure_marshal_VOID__OBJECT, G_TYPE_NONE, 1, GTK_TYPE_STYLE); gal_a11y_e_table_click_to_add_init (); } static void etcta_init (ETableClickToAdd *etcta) { AtkObject *a11y; etcta->one = NULL; etcta->model = NULL; etcta->eth = NULL; etcta->message = NULL; etcta->row = NULL; etcta->text = NULL; etcta->rect = NULL; etcta->selection = e_table_selection_model_new(); g_signal_connect(etcta->selection, "cursor_changed", G_CALLBACK (etcta_cursor_change), etcta); e_canvas_item_set_reflow_callback (GNOME_CANVAS_ITEM (etcta), etcta_reflow); /* create its a11y object at this time if accessibility is enabled*/ if (atk_get_root () != NULL) { a11y = atk_gobject_accessible_for_object (G_OBJECT (etcta)); atk_object_set_name (a11y, _("click to add")); } } /* The colors in this need to be themefied. */ /** * e_table_click_to_add_commit: * @etcta: The %ETableClickToAdd to commit. * * This routine commits the current thing being edited and returns to * just displaying the click to add message. **/ void e_table_click_to_add_commit (ETableClickToAdd *etcta) { if (etcta->row) { e_table_one_commit(E_TABLE_ONE(etcta->one)); etcta_drop_one (etcta); gtk_object_destroy(GTK_OBJECT (etcta->row)); etcta->row = NULL; } create_rect_and_text (etcta); e_canvas_item_move_absolute (etcta->text, 3, 3); }