/* Year view display for gncal * * Copyright (C) 1998 The Free Software Foundation * * Authors: Arturo Espinosa * Federico Mena */ #include #include #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)); }