This makes evolution depend on theme-defined named colors, namely: theme_bg_color theme_base_color theme_fg_color theme_text_color theme_selected_bg_color theme_selected_fg_color theme_unfocused_selected_bg_color theme_unfocused_selected_fg_color If it's not defined, then a fallback color is used, in the worse case one of the fallbacks defined in evolution itself.
856 lines
23 KiB
C
856 lines
23 KiB
C
/*
|
|
* 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.
|
|
*
|
|
* 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 General Public License
|
|
* for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with this program; if not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*
|
|
* Authors:
|
|
* Damon Chaplin <damon@ximian.com>
|
|
* Bolian Yin <bolian.yin@sun.com>
|
|
*
|
|
* Copyright (C) 1999-2008 Novell, Inc. (www.novell.com)
|
|
*
|
|
*/
|
|
|
|
/*
|
|
* ECalendar - displays a table of monthly calendars, allowing highlighting
|
|
* and selection of one or more days. Like GtkCalendar with more features.
|
|
* Most of the functionality is in the ECalendarItem canvas item, though
|
|
* we also add GnomeCanvasWidget buttons to go to the previous/next month and
|
|
* to got to the current day.
|
|
*/
|
|
|
|
#include "e-calendar.h"
|
|
|
|
#include <config.h>
|
|
#include <glib/gi18n.h>
|
|
|
|
#include <gtk/gtk.h>
|
|
#include <libedataserver/libedataserver.h>
|
|
|
|
#include <libgnomecanvas/gnome-canvas-widget.h>
|
|
|
|
#include "e-misc-utils.h"
|
|
|
|
#define E_CALENDAR_SMALL_FONT_PTSIZE 6
|
|
|
|
#define E_CALENDAR_SMALL_FONT \
|
|
"-adobe-utopia-regular-r-normal-*-*-100-*-*-p-*-iso8859-*"
|
|
#define E_CALENDAR_SMALL_FONT_FALLBACK \
|
|
"-adobe-helvetica-medium-r-normal-*-*-80-*-*-p-*-iso8859-*"
|
|
|
|
/* The space between the arrow buttons and the edge of the widget. */
|
|
#define E_CALENDAR_ARROW_BUTTON_X_PAD 2
|
|
#define E_CALENDAR_ARROW_BUTTON_Y_PAD 0
|
|
|
|
/* Vertical padding. The padding above the button includes the space for the
|
|
* horizontal line. */
|
|
#define E_CALENDAR_YPAD_ABOVE_LOWER_BUTTONS 4
|
|
#define E_CALENDAR_YPAD_BELOW_LOWER_BUTTONS 3
|
|
|
|
/* Horizontal padding inside & between buttons. */
|
|
#define E_CALENDAR_IXPAD_BUTTONS 4
|
|
#define E_CALENDAR_XPAD_BUTTONS 8
|
|
|
|
/* The time between steps when the prev/next buttons is pressed, in 1/1000ths
|
|
* of a second, and the number of timeouts we skip before we start
|
|
* automatically moving back/forward. */
|
|
#define E_CALENDAR_AUTO_MOVE_TIMEOUT 150
|
|
#define E_CALENDAR_AUTO_MOVE_TIMEOUT_DELAY 2
|
|
|
|
static void e_calendar_dispose (GObject *object);
|
|
static void e_calendar_realize (GtkWidget *widget);
|
|
static void e_calendar_style_updated (GtkWidget *widget);
|
|
static void e_calendar_get_preferred_width (GtkWidget *widget,
|
|
gint *minimal_width,
|
|
gint *natural_width);
|
|
static void e_calendar_get_preferred_height (GtkWidget *widget,
|
|
gint *minimal_height,
|
|
gint *natural_height);
|
|
static void e_calendar_size_allocate (GtkWidget *widget,
|
|
GtkAllocation *allocation);
|
|
static gint e_calendar_drag_motion (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
guint time);
|
|
static void e_calendar_drag_leave (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
guint time);
|
|
static gboolean e_calendar_button_has_focus (ECalendar *cal);
|
|
static gboolean e_calendar_focus (GtkWidget *widget,
|
|
GtkDirectionType direction);
|
|
|
|
static void e_calendar_on_prev_pressed (ECalendar *cal);
|
|
static void e_calendar_on_prev_released (ECalendar *cal);
|
|
static void e_calendar_on_prev_clicked (ECalendar *cal);
|
|
static void e_calendar_on_next_pressed (ECalendar *cal);
|
|
static void e_calendar_on_next_released (ECalendar *cal);
|
|
static void e_calendar_on_next_clicked (ECalendar *cal);
|
|
static void e_calendar_on_prev_year_pressed (ECalendar *cal);
|
|
static void e_calendar_on_prev_year_released (ECalendar *cal);
|
|
static void e_calendar_on_prev_year_clicked (ECalendar *cal);
|
|
static void e_calendar_on_next_year_pressed (ECalendar *cal);
|
|
static void e_calendar_on_next_year_released (ECalendar *cal);
|
|
static void e_calendar_on_next_year_clicked (ECalendar *cal);
|
|
|
|
static void e_calendar_start_auto_move (ECalendar *cal,
|
|
gboolean moving_forward);
|
|
static gboolean e_calendar_auto_move_handler (gpointer data);
|
|
static void e_calendar_start_auto_move_year (ECalendar *cal,
|
|
gboolean moving_forward);
|
|
static gboolean e_calendar_auto_move_year_handler (gpointer data);
|
|
static void e_calendar_stop_auto_move (ECalendar *cal);
|
|
|
|
G_DEFINE_TYPE (
|
|
ECalendar,
|
|
e_calendar,
|
|
E_TYPE_CANVAS)
|
|
|
|
static void
|
|
e_calendar_class_init (ECalendarClass *class)
|
|
{
|
|
GObjectClass *object_class;
|
|
GtkWidgetClass *widget_class;
|
|
|
|
object_class = (GObjectClass *) class;
|
|
widget_class = (GtkWidgetClass *) class;
|
|
|
|
object_class->dispose = e_calendar_dispose;
|
|
|
|
widget_class->realize = e_calendar_realize;
|
|
widget_class->style_updated = e_calendar_style_updated;
|
|
widget_class->get_preferred_width = e_calendar_get_preferred_width;
|
|
widget_class->get_preferred_height = e_calendar_get_preferred_height;
|
|
widget_class->size_allocate = e_calendar_size_allocate;
|
|
widget_class->drag_motion = e_calendar_drag_motion;
|
|
widget_class->drag_leave = e_calendar_drag_leave;
|
|
widget_class->focus = e_calendar_focus;
|
|
}
|
|
|
|
static void
|
|
e_calendar_init (ECalendar *cal)
|
|
{
|
|
GnomeCanvasGroup *canvas_group;
|
|
PangoFontDescription *small_font_desc;
|
|
PangoContext *pango_context;
|
|
GtkWidget *button, *pixmap;
|
|
AtkObject *a11y;
|
|
|
|
pango_context = gtk_widget_create_pango_context (GTK_WIDGET (cal));
|
|
g_warn_if_fail (pango_context != NULL);
|
|
|
|
/* Create the small font. */
|
|
|
|
small_font_desc = pango_font_description_copy (
|
|
pango_context_get_font_description (pango_context));
|
|
pango_font_description_set_size (
|
|
small_font_desc,
|
|
E_CALENDAR_SMALL_FONT_PTSIZE * PANGO_SCALE);
|
|
|
|
canvas_group = GNOME_CANVAS_GROUP (GNOME_CANVAS (cal)->root);
|
|
|
|
cal->calitem = E_CALENDAR_ITEM (
|
|
gnome_canvas_item_new (
|
|
canvas_group,
|
|
e_calendar_item_get_type (),
|
|
"week_number_font_desc", small_font_desc,
|
|
NULL));
|
|
|
|
pango_font_description_free (small_font_desc);
|
|
g_object_unref (pango_context);
|
|
|
|
/* Create the arrow buttons to move to the previous/next month. */
|
|
button = gtk_button_new ();
|
|
gtk_button_set_relief (GTK_BUTTON (button), GTK_RELIEF_NONE);
|
|
gtk_widget_show (button);
|
|
g_signal_connect_swapped (
|
|
button, "pressed",
|
|
G_CALLBACK (e_calendar_on_prev_pressed), cal);
|
|
g_signal_connect_swapped (
|
|
button, "released",
|
|
G_CALLBACK (e_calendar_on_prev_released), cal);
|
|
g_signal_connect_swapped (
|
|
button, "clicked",
|
|
G_CALLBACK (e_calendar_on_prev_clicked), cal);
|
|
|
|
pixmap = gtk_arrow_new (GTK_ARROW_LEFT, GTK_SHADOW_NONE);
|
|
gtk_widget_show (pixmap);
|
|
gtk_container_add (GTK_CONTAINER (button), pixmap);
|
|
|
|
cal->prev_item = gnome_canvas_item_new (
|
|
canvas_group,
|
|
gnome_canvas_widget_get_type (),
|
|
"widget", button,
|
|
NULL);
|
|
a11y = gtk_widget_get_accessible (button);
|
|
atk_object_set_name (a11y, _("Previous month"));
|
|
|
|
button = gtk_button_new ();
|
|
gtk_button_set_relief (GTK_BUTTON (button), GTK_RELIEF_NONE);
|
|
gtk_widget_show (button);
|
|
g_signal_connect_swapped (
|
|
button, "pressed",
|
|
G_CALLBACK (e_calendar_on_next_pressed), cal);
|
|
g_signal_connect_swapped (
|
|
button, "released",
|
|
G_CALLBACK (e_calendar_on_next_released), cal);
|
|
g_signal_connect_swapped (
|
|
button, "clicked",
|
|
G_CALLBACK (e_calendar_on_next_clicked), cal);
|
|
|
|
pixmap = gtk_arrow_new (GTK_ARROW_RIGHT, GTK_SHADOW_NONE);
|
|
gtk_widget_show (pixmap);
|
|
gtk_container_add (GTK_CONTAINER (button), pixmap);
|
|
|
|
cal->next_item = gnome_canvas_item_new (
|
|
canvas_group,
|
|
gnome_canvas_widget_get_type (),
|
|
"widget", button,
|
|
NULL);
|
|
a11y = gtk_widget_get_accessible (button);
|
|
atk_object_set_name (a11y, _("Next month"));
|
|
|
|
/* Create the arrow buttons to move to the previous/next year. */
|
|
button = gtk_button_new ();
|
|
gtk_button_set_relief (GTK_BUTTON (button), GTK_RELIEF_NONE);
|
|
gtk_widget_show (button);
|
|
g_signal_connect_swapped (
|
|
button, "pressed",
|
|
G_CALLBACK (e_calendar_on_prev_year_pressed), cal);
|
|
g_signal_connect_swapped (
|
|
button, "released",
|
|
G_CALLBACK (e_calendar_on_prev_year_released), cal);
|
|
g_signal_connect_swapped (
|
|
button, "clicked",
|
|
G_CALLBACK (e_calendar_on_prev_year_clicked), cal);
|
|
|
|
pixmap = gtk_arrow_new (GTK_ARROW_LEFT, GTK_SHADOW_NONE);
|
|
gtk_widget_show (pixmap);
|
|
gtk_container_add (GTK_CONTAINER (button), pixmap);
|
|
|
|
cal->prev_item_year = gnome_canvas_item_new (
|
|
canvas_group,
|
|
gnome_canvas_widget_get_type (),
|
|
"widget", button,
|
|
NULL);
|
|
a11y = gtk_widget_get_accessible (button);
|
|
atk_object_set_name (a11y, _("Previous year"));
|
|
|
|
button = gtk_button_new ();
|
|
gtk_button_set_relief (GTK_BUTTON (button), GTK_RELIEF_NONE);
|
|
gtk_widget_show (button);
|
|
g_signal_connect_swapped (
|
|
button, "pressed",
|
|
G_CALLBACK (e_calendar_on_next_year_pressed), cal);
|
|
g_signal_connect_swapped (
|
|
button, "released",
|
|
G_CALLBACK (e_calendar_on_next_year_released), cal);
|
|
g_signal_connect_swapped (
|
|
button, "clicked",
|
|
G_CALLBACK (e_calendar_on_next_year_clicked), cal);
|
|
|
|
pixmap = gtk_arrow_new (GTK_ARROW_RIGHT, GTK_SHADOW_NONE);
|
|
gtk_widget_show (pixmap);
|
|
gtk_container_add (GTK_CONTAINER (button), pixmap);
|
|
|
|
cal->next_item_year = gnome_canvas_item_new (
|
|
canvas_group,
|
|
gnome_canvas_widget_get_type (),
|
|
"widget", button,
|
|
NULL);
|
|
a11y = gtk_widget_get_accessible (button);
|
|
atk_object_set_name (a11y, _("Next year"));
|
|
|
|
cal->min_rows = 1;
|
|
cal->min_cols = 1;
|
|
cal->max_rows = -1;
|
|
cal->max_cols = -1;
|
|
|
|
cal->timeout_id = 0;
|
|
}
|
|
|
|
/**
|
|
* e_calendar_new:
|
|
* @Returns: a new #ECalendar.
|
|
*
|
|
* Creates a new #ECalendar.
|
|
**/
|
|
GtkWidget *
|
|
e_calendar_new (void)
|
|
{
|
|
GtkWidget *cal;
|
|
AtkObject *a11y;
|
|
|
|
cal = g_object_new (e_calendar_get_type (), NULL);
|
|
a11y = gtk_widget_get_accessible (cal);
|
|
atk_object_set_name (a11y, _("Month Calendar"));
|
|
|
|
return cal;
|
|
}
|
|
|
|
static void
|
|
e_calendar_dispose (GObject *object)
|
|
{
|
|
ECalendar *cal;
|
|
|
|
g_return_if_fail (object != NULL);
|
|
g_return_if_fail (E_IS_CALENDAR (object));
|
|
|
|
cal = E_CALENDAR (object);
|
|
|
|
if (cal->timeout_id != 0) {
|
|
g_source_remove (cal->timeout_id);
|
|
cal->timeout_id = 0;
|
|
}
|
|
|
|
/* Chain up to parent's dispose() method. */
|
|
G_OBJECT_CLASS (e_calendar_parent_class)->dispose (object);
|
|
}
|
|
|
|
static void
|
|
e_calendar_update_window_background (GtkWidget *widget)
|
|
{
|
|
GdkWindow *window;
|
|
GdkRGBA bg_bg;
|
|
|
|
e_utils_get_theme_color (widget, "theme_bg_color", E_UTILS_DEFAULT_THEME_BG_COLOR, &bg_bg);
|
|
|
|
/* Set the background of the canvas window to the normal color,
|
|
* or the arrow buttons are not displayed properly. */
|
|
window = gtk_layout_get_bin_window (GTK_LAYOUT (widget));
|
|
gdk_window_set_background_rgba (window, &bg_bg);
|
|
}
|
|
|
|
static void
|
|
e_calendar_realize (GtkWidget *widget)
|
|
{
|
|
(*GTK_WIDGET_CLASS (e_calendar_parent_class)->realize) (widget);
|
|
|
|
e_calendar_update_window_background (widget);
|
|
}
|
|
|
|
static void
|
|
e_calendar_style_updated (GtkWidget *widget)
|
|
{
|
|
ECalendar *e_calendar;
|
|
|
|
e_calendar = E_CALENDAR (widget);
|
|
if (GTK_WIDGET_CLASS (e_calendar_parent_class)->style_updated)
|
|
(*GTK_WIDGET_CLASS (e_calendar_parent_class)->style_updated) (widget);
|
|
|
|
/* Set the background of the canvas window to the normal color,
|
|
* or the arrow buttons are not displayed properly. */
|
|
if (gtk_widget_get_realized (widget))
|
|
e_calendar_update_window_background (widget);
|
|
|
|
e_calendar_item_style_updated (widget, e_calendar->calitem);
|
|
}
|
|
|
|
static void
|
|
e_calendar_get_preferred_width (GtkWidget *widget,
|
|
gint *minimum,
|
|
gint *natural)
|
|
{
|
|
ECalendar *cal;
|
|
GtkBorder padding;
|
|
gint col_width;
|
|
|
|
cal = E_CALENDAR (widget);
|
|
|
|
g_object_get ((cal->calitem), "column_width", &col_width, NULL);
|
|
gtk_style_context_get_padding (gtk_widget_get_style_context (widget), 0, &padding);
|
|
|
|
*minimum = *natural = col_width * cal->min_cols + padding.left * 2;
|
|
}
|
|
|
|
static void
|
|
e_calendar_get_preferred_height (GtkWidget *widget,
|
|
gint *minimum,
|
|
gint *natural)
|
|
{
|
|
ECalendar *cal;
|
|
GtkBorder padding;
|
|
gint row_height;
|
|
|
|
cal = E_CALENDAR (widget);
|
|
|
|
g_object_get ((cal->calitem), "row_height", &row_height, NULL);
|
|
gtk_style_context_get_padding (gtk_widget_get_style_context (widget), 0, &padding);
|
|
|
|
*minimum = *natural = row_height * cal->min_rows + padding.top * 2;
|
|
}
|
|
|
|
static void
|
|
e_calendar_size_allocate (GtkWidget *widget,
|
|
GtkAllocation *allocation)
|
|
{
|
|
ECalendar *cal;
|
|
GtkBorder padding;
|
|
GtkAllocation old_allocation;
|
|
PangoContext *pango_context;
|
|
PangoFontMetrics *font_metrics;
|
|
gdouble old_x2, old_y2, new_x2, new_y2;
|
|
gdouble xthickness, ythickness, arrow_button_size, current_x, month_width;
|
|
gboolean is_rtl;
|
|
|
|
cal = E_CALENDAR (widget);
|
|
gtk_style_context_get_padding (gtk_widget_get_style_context (widget), 0, &padding);
|
|
xthickness = padding.left;
|
|
ythickness = padding.top;
|
|
|
|
(*GTK_WIDGET_CLASS (e_calendar_parent_class)->size_allocate) (widget, allocation);
|
|
|
|
/* Set up Pango prerequisites */
|
|
pango_context = gtk_widget_get_pango_context (widget);
|
|
font_metrics = pango_context_get_metrics (
|
|
pango_context, NULL,
|
|
pango_context_get_language (pango_context));
|
|
|
|
/* Set the scroll region to its allocated size, if changed. */
|
|
gnome_canvas_get_scroll_region (
|
|
GNOME_CANVAS (cal),
|
|
NULL, NULL, &old_x2, &old_y2);
|
|
gtk_widget_get_allocation (widget, &old_allocation);
|
|
new_x2 = old_allocation.width - 1;
|
|
new_y2 = old_allocation.height - 1;
|
|
if (old_x2 != new_x2 || old_y2 != new_y2)
|
|
gnome_canvas_set_scroll_region (
|
|
GNOME_CANVAS (cal),
|
|
0, 0, new_x2, new_y2);
|
|
|
|
/* Take off space for line & buttons if shown. */
|
|
gnome_canvas_item_set (
|
|
GNOME_CANVAS_ITEM (cal->calitem),
|
|
"x1", 0.0,
|
|
"y1", 0.0,
|
|
"x2", new_x2,
|
|
"y2", new_y2,
|
|
NULL);
|
|
|
|
if (cal->calitem->month_width > 0)
|
|
month_width = cal->calitem->month_width;
|
|
else
|
|
month_width = new_x2;
|
|
month_width -= E_CALENDAR_ITEM_MIN_CELL_XPAD + E_CALENDAR_ARROW_BUTTON_X_PAD;
|
|
|
|
/* Position the arrow buttons. */
|
|
arrow_button_size =
|
|
PANGO_PIXELS (pango_font_metrics_get_ascent (font_metrics))
|
|
+ PANGO_PIXELS (pango_font_metrics_get_descent (font_metrics))
|
|
+ E_CALENDAR_ITEM_YPAD_ABOVE_MONTH_NAME
|
|
+ E_CALENDAR_ITEM_YPAD_BELOW_MONTH_NAME
|
|
- E_CALENDAR_ARROW_BUTTON_Y_PAD * 2 - 2;
|
|
|
|
is_rtl = gtk_widget_get_direction (widget) == GTK_TEXT_DIR_RTL;
|
|
current_x = is_rtl ?
|
|
(month_width - 2 * xthickness - E_CALENDAR_ARROW_BUTTON_X_PAD - arrow_button_size) :
|
|
(xthickness);
|
|
|
|
gnome_canvas_item_set (
|
|
cal->prev_item,
|
|
"x", current_x,
|
|
"y", ythickness + E_CALENDAR_ARROW_BUTTON_Y_PAD,
|
|
"width", arrow_button_size,
|
|
"height", arrow_button_size,
|
|
NULL);
|
|
|
|
current_x += (is_rtl ? -1.0 : +1.0) * (cal->calitem->max_month_name_width - xthickness + 2 * arrow_button_size);
|
|
|
|
gnome_canvas_item_set (
|
|
cal->next_item,
|
|
"x", current_x,
|
|
"y", ythickness + E_CALENDAR_ARROW_BUTTON_Y_PAD,
|
|
"width", arrow_button_size,
|
|
"height", arrow_button_size,
|
|
NULL);
|
|
|
|
current_x = is_rtl ?
|
|
(xthickness) :
|
|
(month_width - 2 * xthickness - E_CALENDAR_ARROW_BUTTON_X_PAD - arrow_button_size);
|
|
|
|
gnome_canvas_item_set (
|
|
cal->next_item_year,
|
|
"x", current_x,
|
|
"y", ythickness + E_CALENDAR_ARROW_BUTTON_Y_PAD,
|
|
"width", arrow_button_size,
|
|
"height", arrow_button_size,
|
|
NULL);
|
|
|
|
current_x += (is_rtl ? +1.0 : -1.0) * (cal->calitem->max_digit_width * 5 - xthickness + 2 * arrow_button_size);
|
|
|
|
gnome_canvas_item_set (
|
|
cal->prev_item_year,
|
|
"x", current_x,
|
|
"y", ythickness + E_CALENDAR_ARROW_BUTTON_Y_PAD,
|
|
"width", arrow_button_size,
|
|
"height", arrow_button_size,
|
|
NULL);
|
|
|
|
pango_font_metrics_unref (font_metrics);
|
|
}
|
|
|
|
void
|
|
e_calendar_set_minimum_size (ECalendar *cal,
|
|
gint rows,
|
|
gint cols)
|
|
{
|
|
g_return_if_fail (E_IS_CALENDAR (cal));
|
|
|
|
cal->min_rows = rows;
|
|
cal->min_cols = cols;
|
|
|
|
gnome_canvas_item_set (
|
|
GNOME_CANVAS_ITEM (cal->calitem),
|
|
"minimum_rows", rows,
|
|
"minimum_columns", cols,
|
|
NULL);
|
|
|
|
gtk_widget_queue_resize (GTK_WIDGET (cal));
|
|
}
|
|
|
|
void
|
|
e_calendar_set_maximum_size (ECalendar *cal,
|
|
gint rows,
|
|
gint cols)
|
|
{
|
|
g_return_if_fail (E_IS_CALENDAR (cal));
|
|
|
|
cal->max_rows = rows;
|
|
cal->max_cols = cols;
|
|
|
|
gnome_canvas_item_set (
|
|
GNOME_CANVAS_ITEM (cal->calitem),
|
|
"maximum_rows", rows,
|
|
"maximum_columns", cols,
|
|
NULL);
|
|
|
|
gtk_widget_queue_resize (GTK_WIDGET (cal));
|
|
}
|
|
|
|
/* Returns the border size on each side of the month displays. */
|
|
void
|
|
e_calendar_get_border_size (ECalendar *cal,
|
|
gint *top,
|
|
gint *bottom,
|
|
gint *left,
|
|
gint *right)
|
|
{
|
|
GtkStyleContext *style_context;
|
|
|
|
g_return_if_fail (E_IS_CALENDAR (cal));
|
|
|
|
style_context = gtk_widget_get_style_context (GTK_WIDGET (cal));
|
|
|
|
if (style_context) {
|
|
GtkBorder padding;
|
|
|
|
gtk_style_context_get_padding (style_context, 0, &padding);
|
|
|
|
*top = padding.top;
|
|
*bottom = padding.top;
|
|
*left = padding.left;
|
|
*right = padding.left;
|
|
} else {
|
|
*top = *bottom = *left = *right = 0;
|
|
}
|
|
}
|
|
|
|
static void
|
|
e_calendar_on_prev_pressed (ECalendar *cal)
|
|
{
|
|
e_calendar_start_auto_move (cal, FALSE);
|
|
}
|
|
|
|
static void
|
|
e_calendar_on_next_pressed (ECalendar *cal)
|
|
{
|
|
e_calendar_start_auto_move (cal, TRUE);
|
|
}
|
|
|
|
static void
|
|
e_calendar_on_prev_year_pressed (ECalendar *cal)
|
|
{
|
|
e_calendar_start_auto_move_year (cal, FALSE);
|
|
}
|
|
|
|
static void
|
|
e_calendar_on_next_year_pressed (ECalendar *cal)
|
|
{
|
|
e_calendar_start_auto_move_year (cal, TRUE);
|
|
}
|
|
|
|
static void
|
|
e_calendar_start_auto_move (ECalendar *cal,
|
|
gboolean moving_forward)
|
|
{
|
|
if (cal->timeout_id == 0) {
|
|
cal->timeout_id = e_named_timeout_add (
|
|
E_CALENDAR_AUTO_MOVE_TIMEOUT,
|
|
e_calendar_auto_move_handler, cal);
|
|
}
|
|
|
|
cal->timeout_delay = E_CALENDAR_AUTO_MOVE_TIMEOUT_DELAY;
|
|
cal->moving_forward = moving_forward;
|
|
|
|
}
|
|
|
|
static void
|
|
e_calendar_start_auto_move_year (ECalendar *cal,
|
|
gboolean moving_forward)
|
|
{
|
|
if (cal->timeout_id == 0) {
|
|
cal->timeout_id = e_named_timeout_add (
|
|
E_CALENDAR_AUTO_MOVE_TIMEOUT,
|
|
e_calendar_auto_move_year_handler, cal);
|
|
}
|
|
|
|
cal->timeout_delay = E_CALENDAR_AUTO_MOVE_TIMEOUT_DELAY;
|
|
cal->moving_forward = moving_forward;
|
|
}
|
|
|
|
static gboolean
|
|
e_calendar_auto_move_year_handler (gpointer data)
|
|
{
|
|
ECalendar *cal;
|
|
ECalendarItem *calitem;
|
|
gint offset;
|
|
|
|
g_return_val_if_fail (E_IS_CALENDAR (data), FALSE);
|
|
|
|
cal = E_CALENDAR (data);
|
|
calitem = cal->calitem;
|
|
|
|
if (cal->timeout_delay > 0) {
|
|
cal->timeout_delay--;
|
|
} else {
|
|
offset = cal->moving_forward ? 12 : -12;
|
|
e_calendar_item_set_first_month (
|
|
calitem, calitem->year,
|
|
calitem->month + offset);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static gboolean
|
|
e_calendar_auto_move_handler (gpointer data)
|
|
{
|
|
ECalendar *cal;
|
|
ECalendarItem *calitem;
|
|
gint offset;
|
|
|
|
g_return_val_if_fail (E_IS_CALENDAR (data), FALSE);
|
|
|
|
cal = E_CALENDAR (data);
|
|
calitem = cal->calitem;
|
|
|
|
if (cal->timeout_delay > 0) {
|
|
cal->timeout_delay--;
|
|
} else {
|
|
offset = cal->moving_forward ? 1 : -1;
|
|
e_calendar_item_set_first_month (
|
|
calitem, calitem->year,
|
|
calitem->month + offset);
|
|
}
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static void
|
|
e_calendar_on_prev_released (ECalendar *cal)
|
|
{
|
|
e_calendar_stop_auto_move (cal);
|
|
}
|
|
|
|
static void
|
|
e_calendar_on_next_released (ECalendar *cal)
|
|
{
|
|
e_calendar_stop_auto_move (cal);
|
|
}
|
|
|
|
static void
|
|
e_calendar_on_prev_year_released (ECalendar *cal)
|
|
{
|
|
e_calendar_stop_auto_move (cal);
|
|
}
|
|
|
|
static void
|
|
e_calendar_on_next_year_released (ECalendar *cal)
|
|
{
|
|
e_calendar_stop_auto_move (cal);
|
|
}
|
|
|
|
static void
|
|
e_calendar_stop_auto_move (ECalendar *cal)
|
|
{
|
|
if (cal->timeout_id != 0) {
|
|
g_source_remove (cal->timeout_id);
|
|
cal->timeout_id = 0;
|
|
}
|
|
}
|
|
|
|
static void
|
|
e_calendar_on_prev_clicked (ECalendar *cal)
|
|
{
|
|
e_calendar_item_set_first_month (
|
|
cal->calitem, cal->calitem->year,
|
|
cal->calitem->month - 1);
|
|
}
|
|
|
|
static void
|
|
e_calendar_on_next_clicked (ECalendar *cal)
|
|
{
|
|
e_calendar_item_set_first_month (
|
|
cal->calitem, cal->calitem->year,
|
|
cal->calitem->month + 1);
|
|
}
|
|
|
|
static void
|
|
e_calendar_on_prev_year_clicked (ECalendar *cal)
|
|
{
|
|
e_calendar_item_set_first_month (
|
|
cal->calitem, cal->calitem->year,
|
|
cal->calitem->month - 12);
|
|
}
|
|
|
|
static void
|
|
e_calendar_on_next_year_clicked (ECalendar *cal)
|
|
{
|
|
e_calendar_item_set_first_month (
|
|
cal->calitem, cal->calitem->year,
|
|
cal->calitem->month + 12);
|
|
}
|
|
|
|
static gint
|
|
e_calendar_drag_motion (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
gint x,
|
|
gint y,
|
|
guint time)
|
|
{
|
|
return FALSE;
|
|
}
|
|
|
|
static void
|
|
e_calendar_drag_leave (GtkWidget *widget,
|
|
GdkDragContext *context,
|
|
guint time)
|
|
{
|
|
}
|
|
|
|
static gboolean
|
|
e_calendar_button_has_focus (ECalendar *cal)
|
|
{
|
|
GtkWidget *prev_widget, *next_widget;
|
|
gboolean ret_val;
|
|
|
|
g_return_val_if_fail (E_IS_CALENDAR (cal), FALSE);
|
|
|
|
prev_widget = GNOME_CANVAS_WIDGET (cal->prev_item)->widget;
|
|
next_widget = GNOME_CANVAS_WIDGET (cal->next_item)->widget;
|
|
ret_val = gtk_widget_has_focus (prev_widget) ||
|
|
gtk_widget_has_focus (next_widget);
|
|
return ret_val;
|
|
}
|
|
|
|
static gboolean
|
|
e_calendar_focus (GtkWidget *widget,
|
|
GtkDirectionType direction)
|
|
{
|
|
#define E_CALENDAR_FOCUS_CHILDREN_NUM 5
|
|
ECalendar *cal;
|
|
GnomeCanvas *canvas;
|
|
GnomeCanvasItem *children[E_CALENDAR_FOCUS_CHILDREN_NUM];
|
|
gint focused_index = -1;
|
|
gint index;
|
|
|
|
g_return_val_if_fail (widget != NULL, FALSE);
|
|
g_return_val_if_fail (E_IS_CALENDAR (widget), FALSE);
|
|
cal = E_CALENDAR (widget);
|
|
canvas = GNOME_CANVAS (widget);
|
|
|
|
if (!gtk_widget_get_can_focus (widget))
|
|
return FALSE;
|
|
|
|
children[0] = GNOME_CANVAS_ITEM (cal->calitem);
|
|
children[1] = cal->prev_item;
|
|
children[2] = cal->next_item;
|
|
children[3] = cal->prev_item_year;
|
|
children[4] = cal->next_item_year;
|
|
|
|
/* get current focused item, if e-calendar has had focus */
|
|
if (gtk_widget_has_focus (widget) || e_calendar_button_has_focus (cal))
|
|
for (index = 0; index < E_CALENDAR_FOCUS_CHILDREN_NUM; ++index) {
|
|
if (canvas->focused_item == NULL)
|
|
break;
|
|
|
|
if (children[index] == canvas->focused_item) {
|
|
focused_index = index;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (focused_index == -1)
|
|
if (direction == GTK_DIR_TAB_FORWARD)
|
|
focused_index = 0;
|
|
else
|
|
focused_index = E_CALENDAR_FOCUS_CHILDREN_NUM - 1;
|
|
else
|
|
if (direction == GTK_DIR_TAB_FORWARD)
|
|
++focused_index;
|
|
else
|
|
--focused_index;
|
|
|
|
if (focused_index < 0 ||
|
|
focused_index >= E_CALENDAR_FOCUS_CHILDREN_NUM)
|
|
/* move out of e-calendar */
|
|
return FALSE;
|
|
gnome_canvas_item_grab_focus (children[focused_index]);
|
|
if (GNOME_IS_CANVAS_WIDGET (children[focused_index])) {
|
|
widget = GNOME_CANVAS_WIDGET (children[focused_index])->widget;
|
|
gtk_widget_grab_focus (widget);
|
|
}
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
e_calendar_set_focusable (ECalendar *cal,
|
|
gboolean focusable)
|
|
{
|
|
GtkWidget *widget;
|
|
GtkWidget *prev_widget, *next_widget;
|
|
GtkWidget *toplevel;
|
|
|
|
g_return_if_fail (E_IS_CALENDAR (cal));
|
|
|
|
widget = GTK_WIDGET (cal);
|
|
prev_widget = GNOME_CANVAS_WIDGET (cal->prev_item)->widget;
|
|
next_widget = GNOME_CANVAS_WIDGET (cal->next_item)->widget;
|
|
|
|
if (focusable) {
|
|
gtk_widget_set_can_focus (widget, TRUE);
|
|
gtk_widget_set_can_focus (prev_widget, TRUE);
|
|
gtk_widget_set_can_focus (next_widget, TRUE);
|
|
}
|
|
else {
|
|
if (gtk_widget_has_focus (GTK_WIDGET (cal)) ||
|
|
e_calendar_button_has_focus (cal)) {
|
|
toplevel = gtk_widget_get_toplevel (widget);
|
|
if (toplevel)
|
|
gtk_widget_grab_focus (toplevel);
|
|
}
|
|
gtk_widget_set_can_focus (widget, FALSE);
|
|
gtk_widget_set_can_focus (prev_widget, FALSE);
|
|
gtk_widget_set_can_focus (next_widget, FALSE);
|
|
}
|
|
}
|