/*
*
* 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:
* Bolian Yin <bolian.yin@sun.com>
*
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
*
*/
#include <gtk/gtk.h>
#include <e-util/e-util.h>
#include "ea-calendar-cell.h"
#include "ea-calendar-item.h"
#include "a11y/ea-factory.h"
/* ECalendarCell */
static void e_calendar_cell_class_init (ECalendarCellClass *class);
EA_FACTORY_GOBJECT (EA_TYPE_CALENDAR_CELL, ea_calendar_cell, ea_calendar_cell_new)
GType
e_calendar_cell_get_type (void)
{
static GType type = 0;
if (!type) {
static GTypeInfo tinfo = {
sizeof (ECalendarCellClass),
(GBaseInitFunc) NULL, /* base init */
(GBaseFinalizeFunc) NULL, /* base finalize */
(GClassInitFunc) e_calendar_cell_class_init, /* class init */
(GClassFinalizeFunc) NULL, /* class finalize */
NULL, /* class data */
sizeof (ECalendarCell), /* instance size */
0, /* nb preallocs */
(GInstanceInitFunc) NULL, /* instance init */
NULL /* value table */
};
type = g_type_register_static (G_TYPE_OBJECT,
"ECalendarCell", &tinfo, 0);
}
return type;
}
static void
e_calendar_cell_class_init (ECalendarCellClass *class)
{
EA_SET_FACTORY (e_calendar_cell_get_type (), ea_calendar_cell);
}
ECalendarCell *
e_calendar_cell_new (ECalendarItem *calitem, gint row, gint column)
{
GObject *object;
ECalendarCell *cell;
g_return_val_if_fail (E_IS_CALENDAR_ITEM (calitem), NULL);
object = g_object_new (E_TYPE_CALENDAR_CELL, NULL);
cell = E_CALENDAR_CELL (object);
cell->calitem = calitem;
cell->row = row;
cell->column = column;
#ifdef ACC_DEBUG
g_print ("EvoAcc: e_calendar_cell created %p\n", (gpointer)cell);
#endif
return cell;
}
/* EaCalendarCell */
static void ea_calendar_cell_class_init (EaCalendarCellClass *klass);
static void ea_calendar_cell_init (EaCalendarCell *a11y);
static G_CONST_RETURN gchar * ea_calendar_cell_get_name (AtkObject *accessible);
static G_CONST_RETURN gchar * ea_calendar_cell_get_description (AtkObject *accessible);
static AtkObject * ea_calendar_cell_get_parent (AtkObject *accessible);
static gint ea_calendar_cell_get_index_in_parent (AtkObject *accessible);
static AtkStateSet *ea_calendar_cell_ref_state_set (AtkObject *accessible);
/* component interface */
static void atk_component_interface_init (AtkComponentIface *iface);
static void component_interface_get_extents (AtkComponent *component,
gint *x, gint *y,
gint *width, gint *height,
AtkCoordType coord_type);
static gboolean component_interface_grab_focus (AtkComponent *component);
static gpointer parent_class = NULL;
#ifdef ACC_DEBUG
static gint n_ea_calendar_cell_created = 0, n_ea_calendar_cell_destroyed = 0;
static void ea_calendar_cell_finalize (GObject *object);
#endif
GType
ea_calendar_cell_get_type (void)
{
static GType type = 0;
if (!type) {
static GTypeInfo tinfo = {
sizeof (EaCalendarCellClass),
(GBaseInitFunc) NULL, /* base init */
(GBaseFinalizeFunc) NULL, /* base finalize */
(GClassInitFunc) ea_calendar_cell_class_init, /* class init */
(GClassFinalizeFunc) NULL, /* class finalize */
NULL, /* class data */
sizeof (EaCalendarCell), /* instance size */
0, /* nb preallocs */
(GInstanceInitFunc) ea_calendar_cell_init, /* instance init */
NULL /* value table */
};
static const GInterfaceInfo atk_component_info = {
(GInterfaceInitFunc) atk_component_interface_init,
(GInterfaceFinalizeFunc) NULL,
NULL
};
type = g_type_register_static (ATK_TYPE_GOBJECT_ACCESSIBLE,
"EaCalendarCell", &tinfo, 0);
g_type_add_interface_static (type, ATK_TYPE_COMPONENT,
&atk_component_info);
}
return type;
}
static void
ea_calendar_cell_class_init (EaCalendarCellClass *klass)
{
AtkObjectClass *class = ATK_OBJECT_CLASS (klass);
#ifdef ACC_DEBUG
GObjectClass *gobject_class = G_OBJECT_CLASS (klass);
gobject_class->finalize = ea_calendar_cell_finalize;
#endif
parent_class = g_type_class_peek_parent (klass);
class->get_name = ea_calendar_cell_get_name;
class->get_description = ea_calendar_cell_get_description;
class->get_parent = ea_calendar_cell_get_parent;
class->get_index_in_parent = ea_calendar_cell_get_index_in_parent;
class->ref_state_set = ea_calendar_cell_ref_state_set;
}
static void
ea_calendar_cell_init (EaCalendarCell *a11y)
{
a11y->state_set = atk_state_set_new ();
atk_state_set_add_state (a11y->state_set, ATK_STATE_TRANSIENT);
atk_state_set_add_state (a11y->state_set, ATK_STATE_ENABLED);
atk_state_set_add_state (a11y->state_set, ATK_STATE_SENSITIVE);
atk_state_set_add_state (a11y->state_set, ATK_STATE_SELECTABLE);
atk_state_set_add_state (a11y->state_set, ATK_STATE_SHOWING);
atk_state_set_add_state (a11y->state_set, ATK_STATE_FOCUSABLE);
}
AtkObject*
ea_calendar_cell_new (GObject *obj)
{
gpointer object;
AtkObject *atk_object;
g_return_val_if_fail (E_IS_CALENDAR_CELL (obj), NULL);
object = g_object_new (EA_TYPE_CALENDAR_CELL, NULL);
atk_object = ATK_OBJECT (object);
atk_object_initialize (atk_object, obj);
atk_object->role = ATK_ROLE_TABLE_CELL;
#ifdef ACC_DEBUG
++n_ea_calendar_cell_created;
g_print ("ACC_DEBUG: n_ea_calendar_cell_created = %d\n",
n_ea_calendar_cell_created);
#endif
return atk_object;
}
#ifdef ACC_DEBUG
static void ea_calendar_cell_finalize (GObject *object)
{
G_OBJECT_CLASS (parent_class)->finalize (object);
++n_ea_calendar_cell_destroyed;
g_print ("ACC_DEBUG: n_ea_calendar_cell_destroyed = %d\n",
n_ea_calendar_cell_destroyed);
}
#endif
static G_CONST_RETURN gchar *
ea_calendar_cell_get_name (AtkObject *accessible)
{
GObject *g_obj;
g_return_val_if_fail (EA_IS_CALENDAR_CELL (accessible), NULL);
g_obj = atk_gobject_accessible_get_object (ATK_GOBJECT_ACCESSIBLE (accessible));
if (!g_obj)
/* defunct object*/
return NULL;
if (!accessible->name) {
AtkObject *atk_obj;
EaCalendarItem *ea_calitem;
ECalendarCell *cell;
gint day_index;
gint year, month, day;
gchar buffer[128];
cell = E_CALENDAR_CELL (g_obj);
atk_obj = ea_calendar_cell_get_parent (accessible);
ea_calitem = EA_CALENDAR_ITEM (atk_obj);
day_index = atk_table_get_index_at (ATK_TABLE (ea_calitem),
cell->row, cell->column);
e_calendar_item_get_date_for_offset (cell->calitem, day_index,
&year, &month, &day);
g_snprintf (buffer, 128, "%d-%d-%d", year, month + 1, day);
ATK_OBJECT_CLASS (parent_class)->set_name (accessible, buffer);
}
return accessible->name;
}
static G_CONST_RETURN gchar *
ea_calendar_cell_get_description (AtkObject *accessible)
{
return ea_calendar_cell_get_name (accessible);
}
static AtkObject *
ea_calendar_cell_get_parent (AtkObject *accessible)
{
GObject *g_obj;
ECalendarCell *cell;
ECalendarItem *calitem;
g_return_val_if_fail (EA_IS_CALENDAR_CELL (accessible), NULL);
g_obj = atk_gobject_accessible_get_object (ATK_GOBJECT_ACCESSIBLE (accessible));
if (!g_obj)
/* defunct object*/
return NULL;
cell = E_CALENDAR_CELL (g_obj);
calitem = cell->calitem;
return atk_gobject_accessible_for_object (G_OBJECT (calitem));
}
static gint
ea_calendar_cell_get_index_in_parent (AtkObject *accessible)
{
GObject *g_obj;
ECalendarCell *cell;
AtkObject *parent;
g_return_val_if_fail (EA_IS_CALENDAR_CELL (accessible), -1);
g_obj = atk_gobject_accessible_get_object (ATK_GOBJECT_ACCESSIBLE (accessible));
if (!g_obj)
return -1;
cell = E_CALENDAR_CELL (g_obj);
parent = atk_object_get_parent (accessible);
return atk_table_get_index_at (ATK_TABLE (parent),
cell->row, cell->column);
}
static AtkStateSet *
ea_calendar_cell_ref_state_set (AtkObject *accessible)
{
EaCalendarCell *atk_cell = EA_CALENDAR_CELL (accessible);
g_return_val_if_fail (atk_cell->state_set, NULL);
g_object_ref (atk_cell->state_set);
return atk_cell->state_set;
}
/* Atk Component Interface */
static void
atk_component_interface_init (AtkComponentIface *iface)
{
g_return_if_fail (iface != NULL);
iface->get_extents = component_interface_get_extents;
iface->grab_focus = component_interface_grab_focus;
}
static void
component_interface_get_extents (AtkComponent *component,
gint *x, gint *y, gint *width, gint *height,
AtkCoordType coord_type)
{
GObject *g_obj;
AtkObject *atk_obj, *atk_canvas;
ECalendarCell *cell;
ECalendarItem *calitem;
EaCalendarItem *ea_calitem;
gint day_index;
gint year, month, day;
gint canvas_x, canvas_y, canvas_width, canvas_height;
*x = *y = *width = *height = 0;
g_return_if_fail (EA_IS_CALENDAR_CELL (component));
g_obj = atk_gobject_accessible_get_object (ATK_GOBJECT_ACCESSIBLE (component));
if (!g_obj)
/* defunct object*/
return;
cell = E_CALENDAR_CELL (g_obj);
calitem = cell->calitem;
atk_obj = atk_gobject_accessible_for_object (G_OBJECT (calitem));
ea_calitem = EA_CALENDAR_ITEM (atk_obj);
day_index = atk_table_get_index_at (ATK_TABLE (ea_calitem),
cell->row, cell->column);
e_calendar_item_get_date_for_offset (calitem, day_index,
&year, &month, &day);
if (!e_calendar_item_get_day_extents (calitem,
year, month, day,
x, y, width, height))
return;
atk_canvas = atk_object_get_parent (ATK_OBJECT (ea_calitem));
atk_component_get_extents (ATK_COMPONENT (atk_canvas),
&canvas_x, &canvas_y,
&canvas_width, &canvas_height,
coord_type);
*x += canvas_x;
*y += canvas_y;
}
static gboolean
component_interface_grab_focus (AtkComponent *component)
{
GObject *g_obj;
GtkWidget *toplevel;
AtkObject *ea_calitem;
ECalendarItem *calitem;
EaCalendarCell *a11y;
gint index;
a11y = EA_CALENDAR_CELL (component);
ea_calitem = ea_calendar_cell_get_parent (ATK_OBJECT (a11y));
g_obj = atk_gobject_accessible_get_object (ATK_GOBJECT_ACCESSIBLE (ea_calitem));
calitem = E_CALENDAR_ITEM (g_obj);
index = atk_object_get_index_in_parent (ATK_OBJECT (a11y));
atk_selection_clear_selection (ATK_SELECTION (ea_calitem));
atk_selection_add_selection (ATK_SELECTION (ea_calitem), index);
gtk_widget_grab_focus (GTK_WIDGET (GNOME_CANVAS_ITEM (calitem)->canvas));
toplevel = gtk_widget_get_toplevel (GTK_WIDGET (GNOME_CANVAS_ITEM (calitem)->canvas));
if (toplevel && gtk_widget_is_toplevel (toplevel))
gtk_window_present (GTK_WINDOW (toplevel));
return TRUE;
}