/*
* Copyright 2010 Canonical, Ltd.
*
* This program is free software: you can redistribute it and/or modify it
* under the terms of either or both of the following licenses:
*
* 1) the GNU Lesser General Public License version 3, as published by the
* Free Software Foundation; and/or
* 2) the GNU Lesser General Public License version 2.1, 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 warranties of
* MERCHANTABILITY, SATISFACTORY QUALITY or FITNESS FOR A PARTICULAR
* PURPOSE. See the applicable version of the GNU Lesser General Public
* License for more details.
*
* You should have received a copy of both the GNU Lesser General Public
* License version 3 and version 2.1 along with this program. If not, see
*
*
* Authors:
* Cody Russell
*/
#include
#include "idoactionhelper.h"
#include "idocalendarmenuitem.h"
#include "config.h"
static void ido_calendar_menu_item_finalize (GObject *item);
static void ido_calendar_menu_item_select (GtkMenuItem *item);
static void ido_calendar_menu_item_deselect (GtkMenuItem *item);
static gboolean ido_calendar_menu_item_button_release (GtkWidget *widget,
GdkEventButton *event);
static gboolean ido_calendar_menu_item_button_press (GtkWidget *widget,
GdkEventButton *event);
static gboolean ido_calendar_menu_item_key_press (GtkWidget *widget,
GdkEventKey *event,
gpointer data);
static void ido_calendar_menu_item_send_focus_change (GtkWidget *widget,
gboolean in);
static void calendar_realized_cb (GtkWidget *widget,
IdoCalendarMenuItem *item);
static void calendar_move_focus_cb (GtkWidget *widget,
GtkDirectionType direction,
IdoCalendarMenuItem *item);
static void calendar_month_changed_cb (GtkWidget *widget,
gpointer user_data);
static void calendar_day_selected_double_click_cb (GtkWidget *widget,
gpointer user_data);
static void calendar_day_selected_cb (GtkWidget *widget,
gpointer user_data);
typedef struct {
GtkWidget *box;
GtkWidget *calendar;
GtkWidget *parent;
gboolean selected;
} IdoCalendarMenuItemPrivate;
G_DEFINE_TYPE_WITH_PRIVATE (IdoCalendarMenuItem, ido_calendar_menu_item, GTK_TYPE_MENU_ITEM)
static void
ido_calendar_menu_item_class_init (IdoCalendarMenuItemClass *klass)
{
GObjectClass *gobject_class;
GtkWidgetClass *widget_class;
GtkMenuItemClass *menu_item_class;
gobject_class = G_OBJECT_CLASS (klass);
widget_class = GTK_WIDGET_CLASS (klass);
menu_item_class = GTK_MENU_ITEM_CLASS (klass);
gobject_class->finalize = ido_calendar_menu_item_finalize;
widget_class->button_release_event = ido_calendar_menu_item_button_release;
widget_class->button_press_event = ido_calendar_menu_item_button_press;
menu_item_class->select = ido_calendar_menu_item_select;
menu_item_class->deselect = ido_calendar_menu_item_deselect;
menu_item_class->hide_on_activate = TRUE;
g_signal_new("month-changed", G_TYPE_FROM_CLASS(klass),
G_SIGNAL_RUN_LAST, 0, NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE, 0);
g_signal_new("day-selected", G_TYPE_FROM_CLASS(klass),
G_SIGNAL_RUN_LAST, 0, NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE, 0);
g_signal_new("day-selected-double-click", G_TYPE_FROM_CLASS(klass),
G_SIGNAL_RUN_LAST, 0, NULL, NULL,
g_cclosure_marshal_VOID__VOID,
G_TYPE_NONE, 0);
}
static void
ido_calendar_menu_item_init (IdoCalendarMenuItem *item)
{
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(item);
/* Will be disposed automatically */
priv->calendar = g_object_new (gtk_calendar_get_type (),
NULL);
g_object_add_weak_pointer (G_OBJECT (priv->calendar),
(gpointer*) &priv->calendar);
g_signal_connect (priv->calendar,
"realize",
G_CALLBACK (calendar_realized_cb),
item);
g_signal_connect (priv->calendar,
"move-focus",
G_CALLBACK (calendar_move_focus_cb),
item);
priv->box = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 0);
gtk_box_pack_start (GTK_BOX (priv->box), priv->calendar, TRUE, TRUE, 0);
gtk_container_add (GTK_CONTAINER (item), priv->box);
gtk_widget_show_all (priv->box);
}
static void
ido_calendar_menu_item_finalize (GObject *object)
{
IdoCalendarMenuItem *item = IDO_CALENDAR_MENU_ITEM (object);
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(item);
if (G_IS_OBJECT (priv->calendar))
{
g_object_remove_weak_pointer (G_OBJECT (priv->calendar),
(gpointer*) &priv->calendar);
g_signal_handlers_disconnect_by_data (priv->calendar, item);
}
if (G_IS_OBJECT (priv->parent))
{
g_object_remove_weak_pointer (G_OBJECT (priv->parent),
(gpointer*) &priv->parent);
g_signal_handlers_disconnect_by_data (priv->parent, item);
}
G_OBJECT_CLASS (ido_calendar_menu_item_parent_class)->finalize (object);
}
static void
ido_calendar_menu_item_send_focus_change (GtkWidget *widget,
gboolean in)
{
GdkEvent *event = gdk_event_new (GDK_FOCUS_CHANGE);
g_object_ref (widget);
if (in)
gtk_widget_grab_focus (widget);
event->focus_change.type = GDK_FOCUS_CHANGE;
event->focus_change.window = g_object_ref (gtk_widget_get_window (widget));
event->focus_change.in = in;
gtk_widget_event (widget, event);
g_object_notify (G_OBJECT (widget), "has-focus");
g_object_unref (widget);
gdk_event_free (event);
}
static gboolean
ido_calendar_menu_item_key_press (GtkWidget *widget,
GdkEventKey *event,
gpointer data)
{
IdoCalendarMenuItem *menuitem = IDO_CALENDAR_MENU_ITEM (data);
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
g_return_val_if_fail (IDO_IS_CALENDAR_MENU_ITEM (menuitem), FALSE);
if (priv->selected)
{
GtkWidget *calendar = priv->calendar;
gtk_widget_event (calendar,
((GdkEvent *)(void*)(event)));
if (gtk_widget_get_window (calendar) != NULL)
{
gdk_window_raise (gtk_widget_get_window (calendar));
}
if (!gtk_widget_has_focus (calendar))
{
gtk_widget_grab_focus (calendar);
}
return (event->keyval != GDK_KEY_Return) &&
(event->keyval != GDK_KEY_Escape);
}
return FALSE;
}
static gboolean
ido_calendar_menu_item_button_press (GtkWidget *widget,
GdkEventButton *event)
{
IdoCalendarMenuItem *menuitem = IDO_CALENDAR_MENU_ITEM (widget);
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
GtkWidget *calendar = priv->calendar;
if (event->button == 1) {
if (gtk_widget_get_window (calendar) != NULL) {
gdk_window_raise (gtk_widget_get_window (calendar));
}
if (!gtk_widget_has_focus (calendar)) {
gtk_widget_grab_focus (calendar);
}
GdkEvent * newevent = gdk_event_copy((GdkEvent *)(event));
GList * children = gdk_window_get_children(gtk_widget_get_window(calendar));
GList * child;
gint root_x = event->x_root;
gint root_y = event->y_root;
for (child = children; child != NULL; child = g_list_next(child)) {
gint newx, newy;
gint winx, winy;
GdkWindow * newwindow = (GdkWindow*)child->data;
((GdkEventButton *)newevent)->window = newwindow;
gdk_window_get_origin(newwindow, &winx, &winy);
newx = root_x - winx;
newy = root_y - winy;
if (newx >= 0 && newy >= 0 && newx < gdk_window_get_width(newwindow) && newy < gdk_window_get_height(newwindow)) {
((GdkEventButton *)newevent)->x = newx;
((GdkEventButton *)newevent)->y = newy;
GTK_WIDGET_GET_CLASS(calendar)->button_press_event(GTK_WIDGET(calendar), (GdkEventButton*)newevent);
}
}
((GdkEventButton *)newevent)->window = event->window;
gdk_event_free(newevent);
return TRUE;
}
return FALSE;
}
static gboolean
ido_calendar_menu_item_button_release (GtkWidget *widget,
GdkEventButton *event)
{
IdoCalendarMenuItem *menuitem = IDO_CALENDAR_MENU_ITEM (widget);
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
GtkWidget *calendar = priv->calendar;
GTK_WIDGET_GET_CLASS(calendar)->button_release_event(GTK_WIDGET(calendar), event);
return TRUE;
}
static void
ido_calendar_menu_item_select (GtkMenuItem *item)
{
IdoCalendarMenuItem *menuitem = IDO_CALENDAR_MENU_ITEM (item);
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
priv->selected = TRUE;
ido_calendar_menu_item_send_focus_change (GTK_WIDGET (priv->calendar), TRUE);
}
static void
ido_calendar_menu_item_deselect (GtkMenuItem *item)
{
IdoCalendarMenuItem *menuitem = IDO_CALENDAR_MENU_ITEM (item);
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
priv->selected = FALSE;
ido_calendar_menu_item_send_focus_change (GTK_WIDGET (priv->calendar), FALSE);
}
static void
calendar_realized_cb (GtkWidget *widget,
IdoCalendarMenuItem *item)
{
if (gtk_widget_get_window (widget) != NULL)
{
gdk_window_raise (gtk_widget_get_window (widget));
}
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(item);
priv->parent = gtk_widget_get_parent (GTK_WIDGET (item));
g_object_add_weak_pointer (G_OBJECT (priv->parent),
(gpointer*) &priv->parent);
g_signal_connect (priv->parent,
"key-press-event",
G_CALLBACK (ido_calendar_menu_item_key_press),
item);
g_signal_connect (priv->calendar,
"month-changed",
G_CALLBACK (calendar_month_changed_cb),
item);
g_signal_connect (priv->calendar,
"day-selected",
G_CALLBACK (calendar_day_selected_cb),
item);
g_signal_connect (priv->calendar,
"day-selected-double-click",
G_CALLBACK (calendar_day_selected_double_click_cb),
item);
ido_calendar_menu_item_send_focus_change (widget, TRUE);
}
static void
calendar_move_focus_cb (GtkWidget *widget,
GtkDirectionType direction,
IdoCalendarMenuItem *item)
{
IdoCalendarMenuItem *menuitem = IDO_CALENDAR_MENU_ITEM (widget);
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
ido_calendar_menu_item_send_focus_change (GTK_WIDGET (priv->calendar), FALSE);
g_signal_emit_by_name (item,
"move-focus",
GTK_DIR_TAB_FORWARD);
}
static void
calendar_month_changed_cb (GtkWidget *widget,
gpointer user_data)
{
IdoCalendarMenuItem *item = (IdoCalendarMenuItem *)user_data;
g_signal_emit_by_name (item, "month-changed", NULL);
}
static void
calendar_day_selected_cb (GtkWidget *widget,
gpointer user_data)
{
IdoCalendarMenuItem *item = (IdoCalendarMenuItem *)user_data;
g_signal_emit_by_name (item, "day-selected", NULL);
}
static void
calendar_day_selected_double_click_cb (GtkWidget *widget,
gpointer user_data)
{
IdoCalendarMenuItem *item = (IdoCalendarMenuItem *)user_data;
guint day, month, year;
gtk_calendar_get_date (GTK_CALENDAR (widget), &year, &month, &day);
g_signal_emit_by_name (item, "day-selected-double-click", NULL);
}
/**
* ido_calendar_menu_item_new:
*
* Creates a new #IdoCalendarMenuItem
*
* Return Value: a new #IdoCalendarMenuItem.
**/
GtkWidget *
ido_calendar_menu_item_new (void)
{
return g_object_new (IDO_TYPE_CALENDAR_MENU_ITEM, NULL);
}
/**
* ido_calendar_menu_item_get_calendar:
* @menuitem: A #IdoCalendarMenuItem
*
* Returns the calendar associated with this menu item.
*
* Return Value: (transfer none): The #GtkCalendar used in this item.
*/
GtkWidget *
ido_calendar_menu_item_get_calendar (IdoCalendarMenuItem *menuitem)
{
g_return_val_if_fail (IDO_IS_CALENDAR_MENU_ITEM (menuitem), NULL);
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
return priv->calendar;
}
/**
* ido_calendar_menu_item_mark_day:
* @menuitem: A #IdoCalendarMenuItem
* @day: the day number to unmark between 1 and 31.
*
* Places a visual marker on a particular day.
*
* Return Value: #TRUE
*/
gboolean
ido_calendar_menu_item_mark_day (IdoCalendarMenuItem *menuitem, guint day)
{
g_return_val_if_fail(IDO_IS_CALENDAR_MENU_ITEM(menuitem), FALSE);
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
gtk_calendar_mark_day(GTK_CALENDAR (priv->calendar), day);
return TRUE;
}
/**
* ido_calendar_menu_item_unmark_day:
* @menuitem: A #IdoCalendarMenuItem
* @day: the day number to unmark between 1 and 31.
*
* Removes the visual marker from a particular day.
*
* Return Value: #TRUE
*/
gboolean
ido_calendar_menu_item_unmark_day (IdoCalendarMenuItem *menuitem, guint day)
{
g_return_val_if_fail(IDO_IS_CALENDAR_MENU_ITEM(menuitem), FALSE);
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
gtk_calendar_unmark_day(GTK_CALENDAR (priv->calendar), day);
return TRUE;
}
/**
* ido_calendar_menu_item_clear_marks:
* @menuitem: A #IdoCalendarMenuItem
*
* Remove all visual markers.
*/
void
ido_calendar_menu_item_clear_marks (IdoCalendarMenuItem *menuitem)
{
g_return_if_fail(IDO_IS_CALENDAR_MENU_ITEM(menuitem));
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
gtk_calendar_clear_marks(GTK_CALENDAR (priv->calendar));
}
/**
* ido_calendar_menu_item_set_display_options:
* @menuitem: A #IdoCalendarMenuItem
* @flags: the display options to set
*
* Set the display options for the calendar.
*/
void
ido_calendar_menu_item_set_display_options (IdoCalendarMenuItem *menuitem, GtkCalendarDisplayOptions flags)
{
g_return_if_fail(IDO_IS_CALENDAR_MENU_ITEM(menuitem));
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
gtk_calendar_set_display_options (GTK_CALENDAR (priv->calendar), flags);
}
/**
* ido_calendar_menu_item_get_display_options:
* @menuitem: A #IdoCalendarMenuItem
*
* Get the display options for the calendar.
*
* Return Value: the display options in use
*/
GtkCalendarDisplayOptions
ido_calendar_menu_item_get_display_options (IdoCalendarMenuItem *menuitem)
{
g_return_val_if_fail(IDO_IS_CALENDAR_MENU_ITEM(menuitem), 0);
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
return gtk_calendar_get_display_options (GTK_CALENDAR (priv->calendar));
}
/**
* ido_calendar_menu_item_get_date:
* @menuitem: A #IdoCalendarMenuItem
* @year: (out) (allow-none): location to store the year as a decimal number (e.g. 2011), or #NULL.
* @month: (out) (allow-none): location to store the month number (between 0 and 11), or #NULL.
* @day: (out) (allow-none): location to store the day number (between 1 and 31), or #NULL.
*
* Gets the selected date.
*/
void
ido_calendar_menu_item_get_date (IdoCalendarMenuItem *menuitem,
guint *year,
guint *month,
guint *day) {
g_return_if_fail(IDO_IS_CALENDAR_MENU_ITEM(menuitem));
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
gtk_calendar_get_date (GTK_CALENDAR (priv->calendar), year, month, day);
}
/**
* ido_calendar_menu_item_set_date:
* @menuitem: A #IdoCalendarMenuItem
* @year: the year to show (e.g. 2011).
* @month: a month number (between 0 and 11).
* @day: The day number (between 1 and 31).
*
* Set the date shown on the calendar.
*
* Return Value: #TRUE
*/
gboolean
ido_calendar_menu_item_set_date (IdoCalendarMenuItem *menuitem,
guint year,
guint month,
guint day)
{
guint old_y, old_m, old_d;
g_return_val_if_fail (IDO_IS_CALENDAR_MENU_ITEM(menuitem), FALSE);
IdoCalendarMenuItemPrivate *priv = ido_calendar_menu_item_get_instance_private(menuitem);
ido_calendar_menu_item_get_date (menuitem, &old_y, &old_m, &old_d);
if ((old_y != year) || (old_m != month))
gtk_calendar_select_month (GTK_CALENDAR (priv->calendar), month, year);
if (old_d != day)
gtk_calendar_select_day (GTK_CALENDAR (priv->calendar), day);
return TRUE;
}
/***
****
****
****
***/
static void
activate_current_day (IdoCalendarMenuItem * ido_calendar,
const char * action_name_key)
{
GObject * o;
const char * action_name;
GActionGroup * action_group;
o = G_OBJECT (ido_calendar);
action_name = g_object_get_data (o, action_name_key);
action_group = g_object_get_data (o, "ido-action-group");
if (action_group && action_name)
{
guint y, m, d;
GDateTime * date_time;
GVariant * target;
ido_calendar_menu_item_get_date (ido_calendar, &y, &m, &d);
m++;
date_time = g_date_time_new_local (y, m, d, 9, 0, 0);
target = g_variant_new_int64 (g_date_time_to_unix (date_time));
g_action_group_activate_action (action_group, action_name, target);
g_date_time_unref (date_time);
}
}
static void
on_day_selected (IdoCalendarMenuItem * ido_calendar)
{
activate_current_day (ido_calendar, "ido-selection-action-name");
}
static void
on_day_double_clicked (IdoCalendarMenuItem * ido_calendar)
{
activate_current_day (ido_calendar, "ido-activation-action-name");
}
static void
on_action_state_changed (IdoActionHelper * helper,
GVariant * state,
gpointer unused G_GNUC_UNUSED)
{
GVariant * v;
const char * key;
IdoCalendarMenuItem * ido_calendar;
ido_calendar = IDO_CALENDAR_MENU_ITEM (ido_action_helper_get_widget (helper));
g_return_if_fail (ido_calendar != NULL);
g_return_if_fail (g_variant_is_of_type (state, G_VARIANT_TYPE_DICTIONARY));
/* an int64 representing a time_t indicating which year and month should
be visible in the calendar and which day should be given the cursor. */
key = "calendar-day";
if ((v = g_variant_lookup_value (state, key, G_VARIANT_TYPE_INT64)))
{
int y, m, d;
time_t t;
GDateTime * date_time;
t = g_variant_get_int64 (v);
date_time = g_date_time_new_from_unix_local (t);
g_date_time_get_ymd (date_time, &y, &m, &d);
m--; /* adjust month from GDateTime (1 based) to GtkCalendar (0 based) */
ido_calendar_menu_item_set_date (ido_calendar, y, m, d);
g_date_time_unref (date_time);
g_variant_unref (v);
}
/* a boolean value of whether or not to show the week numbers */
key = "show-week-numbers";
if ((v = g_variant_lookup_value (state, key, G_VARIANT_TYPE_BOOLEAN)))
{
const GtkCalendarDisplayOptions old_flags = ido_calendar_menu_item_get_display_options (ido_calendar);
GtkCalendarDisplayOptions new_flags = old_flags;
if (g_variant_get_boolean (v))
new_flags |= GTK_CALENDAR_SHOW_WEEK_NUMBERS;
else
new_flags &= ~GTK_CALENDAR_SHOW_WEEK_NUMBERS;
if (new_flags != old_flags)
ido_calendar_menu_item_set_display_options (ido_calendar, new_flags);
g_variant_unref (v);
}
/* an array of int32 day-of-months denoting days that have appointments */
key = "appointment-days";
ido_calendar_menu_item_clear_marks (ido_calendar);
if ((v = g_variant_lookup_value (state, key, G_VARIANT_TYPE("ai"))))
{
gint32 day;
GVariantIter iter;
g_variant_iter_init (&iter, v);
while (g_variant_iter_next (&iter, "i", &day))
ido_calendar_menu_item_mark_day (ido_calendar, day);
g_variant_unref (v);
}
}
GtkMenuItem *
ido_calendar_menu_item_new_from_model (GMenuItem * menu_item,
GActionGroup * actions)
{
GObject * o;
GtkWidget * calendar;
IdoCalendarMenuItem * ido_calendar;
gchar * selection_action_name = NULL;
gchar * activation_action_name = NULL;
/* get the select & activate action names */
g_menu_item_get_attribute (menu_item, "action", "s", &selection_action_name);
g_menu_item_get_attribute (menu_item, "activation-action", "s", &activation_action_name);
/* remember the action group & action names so that we can poke them
when user selects and double-clicks */
ido_calendar = IDO_CALENDAR_MENU_ITEM (ido_calendar_menu_item_new ());
o = G_OBJECT (ido_calendar);
g_object_set_data_full (o, "ido-action-group", g_object_ref(actions), g_object_unref);
g_object_set_data_full (o, "ido-selection-action-name", selection_action_name, g_free);
g_object_set_data_full (o, "ido-activation-action-name", activation_action_name, g_free);
calendar = ido_calendar_menu_item_get_calendar (ido_calendar);
g_signal_connect_swapped (calendar, "day-selected",
G_CALLBACK(on_day_selected), ido_calendar);
g_signal_connect_swapped (calendar, "day-selected-double-click",
G_CALLBACK(on_day_double_clicked), ido_calendar);
/* Use an IdoActionHelper for state updates.
Since we have two separate actions for selection & activation,
we'll do the activation & targets logic here in ido-calendar */
if (selection_action_name != NULL)
{
IdoActionHelper * helper;
helper = ido_action_helper_new (GTK_WIDGET(ido_calendar),
actions,
selection_action_name,
NULL);
g_signal_connect (helper, "action-state-changed",
G_CALLBACK (on_action_state_changed), NULL);
g_signal_connect_swapped (ido_calendar, "destroy",
G_CALLBACK (g_object_unref), helper);
}
return GTK_MENU_ITEM (ido_calendar);
}