/* Year view display for gncal
*
* Copyright (C) 1998 The Free Software Foundation
*
* Authors: Arturo Espinosa <arturo@nuclecu.unam.mx>
* Federico Mena <federico@nuclecu.unam.mx>
*/
#include <config.h>
#include <libgnomeui/gnome-canvas-text.h>
#include "year-view.h"
#include "main.h"
#include "timeutil.h"
#define HEAD_SPACING 4 /* Spacing between year heading and months */
#define TITLE_SPACING 2 /* Spacing between title and calendar */
#define SPACING 4 /* Spacing between months */
static void year_view_class_init (YearViewClass *class);
static void year_view_init (YearView *yv);
static void year_view_destroy (GtkObject *object);
static void year_view_size_request (GtkWidget *widget,
GtkRequisition *requisition);
static void year_view_size_allocate (GtkWidget *widget,
GtkAllocation *allocation);
static GnomeCanvas *parent_class;
GtkType
year_view_get_type (void)
{
static GtkType year_view_type = 0;
if (!year_view_type) {
GtkTypeInfo year_view_info = {
"YearView",
sizeof (YearView),
sizeof (YearViewClass),
(GtkClassInitFunc) year_view_class_init,
(GtkObjectInitFunc) year_view_init,
NULL, /* reserved_1 */
NULL, /* reserved_2 */
(GtkClassInitFunc) NULL
};
year_view_type = gtk_type_unique (gnome_canvas_get_type (), &year_view_info);
}
return year_view_type;
}
static void
year_view_class_init (YearViewClass *class)
{
GtkObjectClass *object_class;
GtkWidgetClass *widget_class;
object_class = (GtkObjectClass *) class;
widget_class = (GtkWidgetClass *) class;
parent_class = gtk_type_class (gnome_canvas_get_type ());
object_class->destroy = year_view_destroy;
widget_class->size_request = year_view_size_request;
widget_class->size_allocate = year_view_size_allocate;
}
/* Resizes the year view's child items. This is done in the idle loop for performance (we avoid
* resizing on every size allocation).
*/
static gint
idle_handler (gpointer data)
{
YearView *yv;
double width, height;
double mwidth, mheight;
double h_yofs, m_yofs;
double x, y;
GtkArg arg;
GdkFont *head_font, *title_font;
int i;
yv = data;
/* Get the fonts to get their size later */
arg.name = "font_gdk";
gtk_object_getv (GTK_OBJECT (yv->heading), 1, &arg);
head_font = GTK_VALUE_BOXED (arg);
arg.name = "font_gdk";
gtk_object_getv (GTK_OBJECT (yv->titles[0]), 1, &arg);
title_font = GTK_VALUE_BOXED (arg);
/* Adjust heading */
gnome_canvas_item_set (yv->heading,
"x", (double) yv->canvas.width / 2.0,
"y", (double) HEAD_SPACING,
NULL);
/* Adjust months */
h_yofs = 2 * HEAD_SPACING + head_font->ascent + head_font->descent;
m_yofs = SPACING + title_font->ascent + title_font->descent;
width = (yv->canvas.width + SPACING) / 3.0;
height = (yv->canvas.height - h_yofs + SPACING) / 4.0;
mwidth = (yv->canvas.width - 2 * SPACING) / 3.0;
mheight = (yv->canvas.height - h_yofs - 3 * SPACING - 4 * m_yofs) / 4.0;
for (i = 0; i < 12; i++) {
x = (i % 3) * width;
y = (i / 3) * height + h_yofs;
/* Title */
gnome_canvas_item_set (yv->titles[i],
"x", x + width / 2.0,
"y", y,
NULL);
/* Month item */
gnome_canvas_item_set (yv->mitems[i],
"x", x,
"y", y + m_yofs,
"width", mwidth,
"height", mheight,
NULL);
}
/* Done */
yv->need_resize = FALSE;
return FALSE;
}
/* Marks the year view as needing a resize, which will be performed during the idle loop */
static void
need_resize (YearView *yv)
{
if (yv->need_resize)
return;
yv->need_resize = TRUE;
yv->idle_id = gtk_idle_add (idle_handler, yv);
}
static void
year_view_init (YearView *yv)
{
int i;
char buf[100];
struct tm tm;
memset (&tm, 0, sizeof (tm));
/* Heading */
yv->heading = gnome_canvas_item_new (GNOME_CANVAS_GROUP (yv->canvas.root),
gnome_canvas_text_get_type (),
"anchor", GTK_ANCHOR_N,
"font", "-*-helvetica-bold-r-normal--14-*-*-*-*-*-iso8859-1",
"fill_color", "black",
NULL);
/* Months */
for (i = 0; i < 12; i++) {
/* Title */
strftime (buf, 100, "%B", &tm);
tm.tm_mon++;
yv->titles[i] = gnome_canvas_item_new (GNOME_CANVAS_GROUP (yv->canvas.root),
gnome_canvas_text_get_type (),
"text", buf,
"anchor", GTK_ANCHOR_N,
"font", "-*-helvetica-bold-r-normal--12-*-*-*-*-*-iso8859-1",
"fill_color", "black",
NULL);
/* Month item */
yv->mitems[i] = gnome_month_item_new (GNOME_CANVAS_GROUP (yv->canvas.root));
gnome_canvas_item_set (yv->mitems[i],
"anchor", GTK_ANCHOR_NW,
"start_on_monday", week_starts_on_monday,
"heading_color", "white",
NULL);
}
/* We will need to resize the items when we paint for the first time */
yv->idle_id = -1;
need_resize (yv);
}
static void
year_view_destroy (GtkObject *object)
{
YearView *yv;
g_return_if_fail (object != NULL);
g_return_if_fail (IS_YEAR_VIEW (object));
yv = YEAR_VIEW (object);
if (yv->need_resize) {
yv->need_resize = FALSE;
gtk_idle_remove (yv->idle_id);
}
if (GTK_OBJECT_CLASS (parent_class)->destroy)
(* GTK_OBJECT_CLASS (parent_class)->destroy) (object);
}
GtkWidget *
year_view_new (GnomeCalendar *calendar, time_t year)
{
YearView *yv;
g_return_val_if_fail (calendar != NULL, NULL);
g_return_val_if_fail (GNOME_IS_CALENDAR (calendar), NULL);
yv = gtk_type_new (year_view_get_type ());
yv->calendar = calendar;
year_view_set (yv, year);
return GTK_WIDGET (yv);
}
static void
year_view_size_request (GtkWidget *widget, GtkRequisition *requisition)
{
YearView *yv;
g_return_if_fail (widget != NULL);
g_return_if_fail (IS_YEAR_VIEW (widget));
g_return_if_fail (requisition != NULL);
yv = YEAR_VIEW (widget);
if (GTK_WIDGET_CLASS (parent_class)->size_request)
(* GTK_WIDGET_CLASS (parent_class)->size_request) (widget, requisition);
requisition->width = 200;
requisition->height = 150;
}
static void
year_view_size_allocate (GtkWidget *widget, GtkAllocation *allocation)
{
YearView *yv;
g_return_if_fail (widget != NULL);
g_return_if_fail (IS_YEAR_VIEW (widget));
g_return_if_fail (allocation != NULL);
yv = YEAR_VIEW (widget);
if (GTK_WIDGET_CLASS (parent_class)->size_allocate)
(* GTK_WIDGET_CLASS (parent_class)->size_allocate) (widget, allocation);
gnome_canvas_set_scroll_region (GNOME_CANVAS (yv), 0, 0, allocation->width, allocation->height);
need_resize (yv);
}
void
year_view_update (YearView *yv, iCalObject *object, int flags)
{
g_return_if_fail (yv != NULL);
g_return_if_fail (IS_YEAR_VIEW (yv));
/* If only the summary changed, we don't care */
if (object && ((flags & CHANGE_SUMMARY) == flags))
return;
year_view_set (yv, time_year_begin (yv->year));
}
/* Unmarks all the days in the year view by setting their boxes and labels to the default colors */
static void
unmark_days (YearView *yv)
{
GnomeCanvasItem *item;
int i, j;
for (i = 0; i < 12; i++)
for (j = 0; j < 42; j++) {
/* Box */
item = gnome_month_item_num2child (GNOME_MONTH_ITEM (yv->mitems[i]),
GNOME_MONTH_ITEM_DAY_BOX + j);
gnome_canvas_item_set (item,
"fill_color", "#d6d6d6d6d6d6",
NULL);
/* Label */
item = gnome_month_item_num2child (GNOME_MONTH_ITEM (yv->mitems[i]),
GNOME_MONTH_ITEM_DAY_LABEL + j);
gnome_canvas_item_set (item,
"fill_color", "black",
NULL);
}
}
/* Marks all the days that fall into the specified time span */
static void
mark_event (YearView *yv, time_t start, time_t end)
{
time_t t;
struct tm tm;
int day_index;
GnomeCanvasItem *mitem, *item;
tm = *localtime (&start);
end = time_end_of_day (end);
for (t = start; t < end; t += 60 * 60 * 24) {
mktime (&tm); /* normalize the time */
/* We need this comparison because an event may span more than one year (!).
* Yes, this is not the most efficient way of doing this (we could just clip
* the event to the current year), but it will do for now.
*/
if ((tm.tm_year + 1900) == yv->year) {
/* Figure out the month item and day index that correspond to this time */
mitem = yv->mitems[tm.tm_mon];
day_index = gnome_month_item_day2index (GNOME_MONTH_ITEM (mitem), tm.tm_mday);
g_assert (day_index != -1);
/* Mark the day box */
item = gnome_month_item_num2child (GNOME_MONTH_ITEM (mitem),
GNOME_MONTH_ITEM_DAY_BOX + day_index);
gnome_canvas_item_set (item,
"fill_color", "tan",
NULL);
/* Mark the day label */
item = gnome_month_item_num2child (GNOME_MONTH_ITEM (mitem),
GNOME_MONTH_ITEM_DAY_LABEL + day_index);
gnome_canvas_item_set (item,
"fill_color", "black",
NULL);
}
/* Next day */
tm.tm_mday++;
}
}
/* Queries the calendar for all the events in the current year and marks the days that have at least
* one event in them.
*/
static void
mark_days (YearView *yv)
{
time_t year_begin, year_end;
GList *list, *l;
CalendarObject *co;
year_begin = time_year_begin (yv->year);
year_end = time_year_end (yv->year);
list = calendar_get_events_in_range (yv->calendar->cal, year_begin, year_end);
for (l = list; l; l = l->next) {
co = l->data;
mark_event (yv, co->ev_start, co->ev_end);
}
calendar_destroy_event_list (list);
}
void
year_view_set (YearView *yv, time_t year)
{
struct tm *tm;
char buf[100];
int i;
g_return_if_fail (yv != NULL);
g_return_if_fail (IS_YEAR_VIEW (yv));
tm = localtime (&year);
yv->year = tm->tm_year + 1900;
/* Heading */
sprintf (buf, "%d", yv->year);
gnome_canvas_item_set (yv->heading,
"text", buf,
NULL);
/* Months */
for (i = 0; i < 12; i++)
gnome_canvas_item_set (yv->mitems[i],
"year", yv->year,
"month", i,
NULL);
unmark_days (yv);
mark_days (yv);
}
void
year_view_time_format_changed (YearView *yv)
{
int i;
g_return_if_fail (yv != NULL);
g_return_if_fail (IS_YEAR_VIEW (yv));
for (i = 0; i < 12; i++)
gnome_canvas_item_set (yv->mitems[i],
"start_on_monday", week_starts_on_monday,
NULL);
year_view_set (yv, time_year_begin (yv->year));
}