[gnome-calendar/wip/gbsneto/gcal-event: 1/8] event: add a wrapper around events
- From: Georges Basile Stavracas Neto <gbsneto src gnome org>
- To: commits-list gnome org
- Cc:
- Subject: [gnome-calendar/wip/gbsneto/gcal-event: 1/8] event: add a wrapper around events
- Date: Sat, 13 Feb 2016 17:34:32 +0000 (UTC)
commit 8b6185a2ba1bd261ee817df5aa37ce2e23132e83
Author: Georges Basile Stavracas Neto <georges stavracas gmail com>
Date: Sun Feb 7 12:32:03 2016 -0200
event: add a wrapper around events
Instead of relying on confusing structures, raw data
and direct access to components, add a wrapper for
events.
src/Makefile.am | 2 +
src/gcal-event.c | 1139 ++++++++++++++++++++++++++++++++++++++++++++++++++++++
src/gcal-event.h | 98 +++++
src/gcal-utils.c | 47 +++
src/gcal-utils.h | 5 +
5 files changed, 1291 insertions(+), 0 deletions(-)
---
diff --git a/src/Makefile.am b/src/Makefile.am
index 9f1fb5a..d465ece 100644
--- a/src/Makefile.am
+++ b/src/Makefile.am
@@ -34,6 +34,8 @@ gnome_calendar_SOURCES = \
gcal-date-selector.h \
gcal-edit-dialog.c \
gcal-edit-dialog.h \
+ gcal-event.c \
+ gcal-event.h \
gcal-event-widget.c \
gcal-event-widget.h \
gcal-manager.c \
diff --git a/src/gcal-event.c b/src/gcal-event.c
new file mode 100644
index 0000000..a900d93
--- /dev/null
+++ b/src/gcal-event.c
@@ -0,0 +1,1139 @@
+/* gcal-event.c
+ *
+ * Copyright (C) 2016 Georges Basile Stavracas Neto <georges stavracas gmail com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * 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 General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#include "gcal-event.h"
+#include "gcal-utils.h"
+
+struct _GcalEvent
+{
+ GObject parent;
+
+ gchar *uid;
+
+ /*
+ * The description is cached in the class because it
+ * may come as a GSList of descriptions, in which
+ * case we merge them and cache it here.
+ */
+ gchar *description;
+
+ GTimeZone *timezone;
+ GDateTime *dt_start;
+ GDateTime *dt_end;
+
+ GdkRGBA *color;
+ GBinding *color_binding;
+
+ gboolean all_day : 1;
+
+ ECalComponent *component;
+ ESource *source;
+};
+
+G_DEFINE_TYPE (GcalEvent, gcal_event, G_TYPE_OBJECT)
+
+enum {
+ PROP_0,
+ PROP_ALL_DAY,
+ PROP_COLOR,
+ PROP_COMPONENT,
+ PROP_DESCRIPTION,
+ PROP_DATE_END,
+ PROP_DATE_START,
+ PROP_LOCATION,
+ PROP_SOURCE,
+ PROP_SUMMARY,
+ PROP_TIMEZONE,
+ PROP_UID,
+ N_PROPS
+};
+
+static ECalComponentDateTime*
+build_component_from_datetime (GDateTime *dt)
+{
+ ECalComponentDateTime *comp_dt;
+ const gchar *tzid;
+
+ if (!dt)
+ return NULL;
+
+ comp_dt = g_new0 (ECalComponentDateTime, 1);
+ comp_dt->value = NULL;
+ comp_dt->tzid = NULL;
+
+ tzid = g_date_time_get_timezone_abbreviation (dt);
+
+ comp_dt->value = datetime_to_icaltime (dt);
+ comp_dt->tzid = g_strdup (tzid ? tzid : "UTC");
+
+ return comp_dt;
+}
+
+static gboolean
+string_to_color (GBinding *binding,
+ const GValue *from_value,
+ GValue *to_value,
+ gpointer user_data)
+{
+ GdkRGBA color;
+
+ if (!gdk_rgba_parse (&color, g_value_get_string (from_value)))
+ gdk_rgba_parse (&color, "#ffffff"); /* calendar default colour */
+
+ g_value_set_boxed (to_value, &color);
+
+ return TRUE;
+}
+
+static void
+gcal_event_update_uid_internal (GcalEvent *self)
+{
+ ECalComponentId *id;
+ const gchar *source_id;
+
+ /* Setup event uid */
+ source_id = self->source ? e_source_get_uid (self->source) : "";
+ id = e_cal_component_get_id (self->component);
+
+ if (id->rid != NULL)
+ {
+ self->uid = g_strdup_printf ("%s:%s:%s",
+ source_id,
+ id->uid,
+ id->rid);
+ }
+ else
+ {
+ self->uid = g_strdup_printf ("%s:%s",
+ source_id,
+ id->uid);
+ }
+
+ g_object_notify (G_OBJECT (self), "uid");
+}
+
+static void
+gcal_event_set_component_internal (GcalEvent *self,
+ ECalComponent *component)
+{
+ if (g_set_object (&self->component, component))
+ {
+ ECalComponentDateTime start;
+ ECalComponentDateTime end;
+ GDateTime *date_start;
+ GDateTime *date_end;
+ GTimeZone *tz;
+ gchar *description;
+
+ /* Setup start date */
+ e_cal_component_get_dtstart (component, &start);
+ date_start = icaltime_to_datetime (start.value, &tz);
+
+ gcal_event_set_date_start (self, date_start);
+
+ /* Setup end date */
+ e_cal_component_get_dtend (component, &end);
+ date_end = icaltime_to_datetime (end.value, NULL);
+
+ gcal_event_set_date_end (self, date_end);
+
+ /*
+ * Setup timezone. We use the timezone from the start
+ * date as the timezone of the event, and the end date
+ * is adjusted if necessary.
+ */
+ gcal_event_set_timezone (self, tz);
+
+ /* Setup all day */
+ gcal_event_set_all_day (self, start.value->is_date || end.value->is_date);
+
+ /* Setup description */
+ description = get_desc_from_component (component, "\n\n");
+ gcal_event_set_description (self, description);
+
+ /* Setup UID */
+ gcal_event_update_uid_internal (self);
+
+ g_object_notify (G_OBJECT (self), "component");
+ g_object_notify (G_OBJECT (self), "location");
+ g_object_notify (G_OBJECT (self), "summary");
+
+ g_clear_pointer (&date_start, g_date_time_unref);
+ g_clear_pointer (&date_end, g_date_time_unref);
+ g_clear_pointer (&description, g_free);
+ }
+}
+
+static void
+gcal_event_finalize (GObject *object)
+{
+ GcalEvent *self = (GcalEvent *)object;
+
+ g_clear_pointer (&self->dt_start, g_date_time_unref);
+ g_clear_pointer (&self->dt_end, g_date_time_unref);
+ g_clear_pointer (&self->timezone, g_time_zone_unref);
+ g_clear_pointer (&self->description, g_free);
+ g_clear_object (&self->component);
+ g_clear_object (&self->source);
+
+ G_OBJECT_CLASS (gcal_event_parent_class)->finalize (object);
+}
+
+static void
+gcal_event_get_property (GObject *object,
+ guint prop_id,
+ GValue *value,
+ GParamSpec *pspec)
+{
+ GcalEvent *self = GCAL_EVENT (object);
+
+ switch (prop_id)
+ {
+ case PROP_ALL_DAY:
+ g_value_set_boolean (value, self->all_day);
+ break;
+
+ case PROP_COLOR:
+ g_value_set_boxed (value, self->color);
+ break;
+
+ case PROP_COMPONENT:
+ g_value_set_object (value, self->component);
+ break;
+
+ case PROP_DATE_END:
+ g_value_set_boxed (value, gcal_event_get_date_end (self));
+ break;
+
+ case PROP_DATE_START:
+ g_value_set_boxed (value, self->dt_start);
+ break;
+
+ case PROP_DESCRIPTION:
+ g_value_set_string (value, gcal_event_get_description (self));
+ break;
+
+ case PROP_LOCATION:
+ g_value_set_string (value, gcal_event_get_location (self));
+ break;
+
+ case PROP_SOURCE:
+ g_value_set_object (value, self->source);
+ break;
+
+ case PROP_SUMMARY:
+ g_value_set_string (value, gcal_event_get_summary (self));
+ break;
+
+ case PROP_TIMEZONE:
+ g_value_set_boxed (value, self->timezone);
+ break;
+
+ case PROP_UID:
+ g_value_set_string (value, self->uid);
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+gcal_event_set_property (GObject *object,
+ guint prop_id,
+ const GValue *value,
+ GParamSpec *pspec)
+{
+ GcalEvent *self = GCAL_EVENT (object);
+
+ switch (prop_id)
+ {
+ case PROP_ALL_DAY:
+ gcal_event_set_all_day (self, g_value_get_boolean (value));
+ break;
+
+ case PROP_COLOR:
+ gcal_event_set_color (self, g_value_get_boxed (value));
+ break;
+
+ case PROP_COMPONENT:
+ gcal_event_set_component_internal (self, g_value_get_object (value));
+ break;
+
+ case PROP_DATE_END:
+ gcal_event_set_date_end (self, g_value_get_boxed (value));
+ break;
+
+ case PROP_DATE_START:
+ gcal_event_set_date_start (self, g_value_get_boxed (value));
+ break;
+
+ case PROP_DESCRIPTION:
+ gcal_event_set_description (self, g_value_get_string (value));
+ break;
+
+ case PROP_LOCATION:
+ gcal_event_set_location (self, g_value_get_string (value));
+ break;
+
+ case PROP_SOURCE:
+ gcal_event_set_source (self, g_value_get_object (value));
+ break;
+
+ case PROP_SUMMARY:
+ gcal_event_set_summary (self, g_value_get_string (value));
+ break;
+
+ case PROP_TIMEZONE:
+ gcal_event_set_timezone (self, g_value_get_boxed (value));
+ break;
+
+ default:
+ G_OBJECT_WARN_INVALID_PROPERTY_ID (object, prop_id, pspec);
+ }
+}
+
+static void
+gcal_event_class_init (GcalEventClass *klass)
+{
+ GObjectClass *object_class = G_OBJECT_CLASS (klass);
+
+ object_class->finalize = gcal_event_finalize;
+ object_class->get_property = gcal_event_get_property;
+ object_class->set_property = gcal_event_set_property;
+
+ /**
+ * GcalEvent::all-day:
+ *
+ * Whether the event is all day or not.
+ */
+ g_object_class_install_property (object_class,
+ PROP_ALL_DAY,
+ g_param_spec_boolean ("all-day",
+ "If event is all day",
+ "Whether the event is all day or not",
+ FALSE,
+ G_PARAM_READWRITE));
+
+ /**
+ * GcalEvent::color:
+ *
+ * The color of the event.
+ */
+ g_object_class_install_property (object_class,
+ PROP_COLOR,
+ g_param_spec_boxed ("color",
+ "Color of the event",
+ "The color of the event",
+ GDK_TYPE_RGBA,
+ G_PARAM_READWRITE));
+
+ /**
+ * GcalEvent::component:
+ *
+ * The #ECalComponent of this event.
+ */
+ g_object_class_install_property (object_class,
+ PROP_COMPONENT,
+ g_param_spec_object ("component",
+ "Component",
+ "The ECalComponent of the event",
+ E_TYPE_CAL_COMPONENT,
+ G_PARAM_READWRITE | G_PARAM_CONSTRUCT_ONLY));
+
+ /**
+ * GcalEvent::date-end:
+ *
+ * The end date of the event.
+ */
+ g_object_class_install_property (object_class,
+ PROP_DATE_END,
+ g_param_spec_boxed ("date-end",
+ "End date of the event",
+ "The end date of the event",
+ G_TYPE_DATE_TIME,
+ G_PARAM_READWRITE));
+
+ /**
+ * GcalEvent::date-start:
+ *
+ * The start date of the event.
+ */
+ g_object_class_install_property (object_class,
+ PROP_DATE_START,
+ g_param_spec_boxed ("date-start",
+ "Start date of the event",
+ "The start date of the event",
+ G_TYPE_DATE_TIME,
+ G_PARAM_READWRITE));
+
+ /**
+ * GcalEvent::description:
+ *
+ * The description of the event.
+ */
+ g_object_class_install_property (object_class,
+ PROP_DESCRIPTION,
+ g_param_spec_string ("description",
+ "Description of the event",
+ "The description of the event",
+ "",
+ G_PARAM_READWRITE));
+
+ /**
+ * GcalEvent::location:
+ *
+ * The location of the event.
+ */
+ g_object_class_install_property (object_class,
+ PROP_LOCATION,
+ g_param_spec_string ("location",
+ "Location of the event",
+ "The location of the event",
+ "",
+ G_PARAM_READWRITE));
+
+ /**
+ * GcalEvent::source:
+ *
+ * The #ESource this event belongs to.
+ */
+ g_object_class_install_property (object_class,
+ PROP_SOURCE,
+ g_param_spec_object ("source",
+ "ESource",
+ "The ESource this event belongs to",
+ E_TYPE_SOURCE,
+ G_PARAM_READWRITE | G_PARAM_CONSTRUCT));
+
+ /**
+ * GcalEvent::summary:
+ *
+ * The summary of the event.
+ */
+ g_object_class_install_property (object_class,
+ PROP_SUMMARY,
+ g_param_spec_string ("summary",
+ "Summary of the event",
+ "The summary of the event",
+ "",
+ G_PARAM_READWRITE));
+
+ /**
+ * GcalEvent::timezone:
+ *
+ * The timezone of the event.
+ */
+ g_object_class_install_property (object_class,
+ PROP_TIMEZONE,
+ g_param_spec_boxed ("timezone",
+ "Timezone of the event",
+ "The timezone of the event",
+ G_TYPE_TIME_ZONE,
+ G_PARAM_READWRITE));
+
+ /**
+ * GcalEvent::uid:
+ *
+ * The unique identifier of the event.
+ */
+ g_object_class_install_property (object_class,
+ PROP_UID,
+ g_param_spec_string ("uid",
+ "Identifier of the event",
+ "The unique identifier of the event",
+ "",
+ G_PARAM_READABLE));
+}
+
+static void
+gcal_event_init (GcalEvent *self)
+{
+ GdkRGBA rgba;
+
+ /* Default color of events */
+ gdk_rgba_parse (&rgba, "#ffffff");
+ self->color = gdk_rgba_copy (&rgba);
+}
+
+/**
+ * gcal_event_new:
+ * @source: (nullable): an #ESource
+ * @component: a #ECalComponent
+ *
+ * Creates a new event which belongs to @source and
+ * is represented by @component. New events will have
+ * a %NULL @source.
+ *
+ * Returns: (transfer full): a #GcalEvent
+ */
+GcalEvent*
+gcal_event_new (ESource *source,
+ ECalComponent *component)
+{
+ return g_object_new (GCAL_TYPE_EVENT,
+ "source", source,
+ "component", component,
+ NULL);
+}
+
+/**
+ * gcal_event_get_all_day:
+ * @self: a #GcalEvent
+ *
+ * Retrieves whether the event is all day or not.
+ *
+ * Returns: %TRUE if @self is all day, %FALSE otherwise.
+ */
+gboolean
+gcal_event_get_all_day (GcalEvent *self)
+{
+ g_return_val_if_fail (GCAL_IS_EVENT (self), FALSE);
+
+ return self->all_day;
+}
+
+/**
+ * gcal_event_get_color:
+ * @self: a #GcalEvent
+ *
+ * Retrieves the color of the event.
+ *
+ * Returns: (transfer none): a #GdkRGBA
+ */
+GdkRGBA*
+gcal_event_get_color (GcalEvent *self)
+{
+ g_return_val_if_fail (GCAL_IS_EVENT (self), NULL);
+
+ return self->color;
+}
+
+/**
+ * gcal_event_set_color:
+ * @self: a #GcalEvent
+ * @color: a #GdkRGBA
+ *
+ * Sets the color of the event, which is passed to the
+ * parent @ESource.
+ */
+void
+gcal_event_set_color (GcalEvent *self,
+ GdkRGBA *color)
+{
+ g_return_if_fail (GCAL_IS_EVENT (self));
+
+ if (!gdk_rgba_equal (self->color, color))
+ {
+ g_clear_pointer (&self->color, gdk_rgba_free);
+ self->color = gdk_rgba_copy (color);
+
+ g_object_notify (G_OBJECT (self), "color");
+ }
+}
+
+/**
+ * gcal_event_get_component:
+ * @self: a #GcalEvent
+ *
+ * Retrieves the internal component of the event.
+ *
+ * Returns: (transfer none): an #ECalComponent.
+ */
+ECalComponent*
+gcal_event_get_component (GcalEvent *self)
+{
+ g_return_val_if_fail (GCAL_IS_EVENT (self), NULL);
+
+ return self->component;
+}
+
+/**
+ * gcal_event_set_all_day:
+ * @self: a #GcalEvent
+ * @all_day: whether the event is all day or not
+ *
+ * Sets the @GcalEvent::all-day property.
+ */
+void
+gcal_event_set_all_day (GcalEvent *self,
+ gboolean all_day)
+{
+ g_return_if_fail (GCAL_IS_EVENT (self));
+
+ if (self->all_day != !!all_day)
+ {
+ self->all_day = !!all_day;
+
+ g_object_notify (G_OBJECT (self), "all-day");
+ }
+}
+
+/**
+ * gcal_event_get_date_end:
+ * @self: a #GcalEvent
+ *
+ * Retrieves the end date of @self.
+ *
+ * Returns: (transfer none): a #GDateTime.
+ */
+GDateTime*
+gcal_event_get_date_end (GcalEvent *self)
+{
+ g_return_val_if_fail (GCAL_IS_EVENT (self), NULL);
+
+ /* End date can be NULL, meaning dt_end is the same as dt_start */
+ return self->dt_end ? self->dt_end : self->dt_start;
+}
+
+/**
+ * gcal_event_set_date_end:
+ * @self: a #GcalEvent
+ * @dt: a #GDateTime
+ *
+ * Sets the end date as @dt.
+ */
+void
+gcal_event_set_date_end (GcalEvent *self,
+ GDateTime *dt)
+{
+ g_return_if_fail (GCAL_IS_EVENT (self));
+
+ if (self->dt_end != dt &&
+ (!self->dt_end || !dt ||
+ (self->dt_end && dt && !g_date_time_equal (self->dt_end, dt))))
+ {
+ ECalComponentDateTime *component_dt;
+
+ g_clear_pointer (&self->dt_end, g_date_time_unref);
+ self->dt_end = g_date_time_ref (dt);
+
+ /* Setup the ECalComponent's datetime value */
+ component_dt = build_component_from_datetime (dt);
+
+ e_cal_component_set_dtend (self->component, component_dt);
+ e_cal_component_commit_sequence (self->component);
+
+ g_object_notify (G_OBJECT (self), "date-end");
+
+ g_clear_pointer (&component_dt, e_cal_component_free_datetime);
+ }
+}
+
+/**
+ * gcal_event_get_date_start:
+ * @self: a #GcalEvent
+ *
+ * Retrieves the start date of @self.
+ *
+ * Returns: (transfer none): a #GDateTime.
+ */
+GDateTime*
+gcal_event_get_date_start (GcalEvent *self)
+{
+ g_return_val_if_fail (GCAL_IS_EVENT (self), NULL);
+
+ return self->dt_start;
+}
+
+/**
+ * gcal_event_set_date_start:
+ * @self: a #GcalEvent
+ * @dt: a #GDateTime
+ *
+ * Sets the start date as @dt.
+ */
+void
+gcal_event_set_date_start (GcalEvent *self,
+ GDateTime *dt)
+{
+ g_return_if_fail (GCAL_IS_EVENT (self));
+
+ if (self->dt_start != dt &&
+ (!self->dt_start || !dt ||
+ (self->dt_start && dt && !g_date_time_equal (self->dt_start, dt))))
+ {
+ ECalComponentDateTime *component_dt;
+
+ g_clear_pointer (&self->dt_start, g_date_time_unref);
+ self->dt_start = g_date_time_ref (dt);
+
+ /* Setup the ECalComponent's datetime value */
+ component_dt = build_component_from_datetime (dt);
+
+ e_cal_component_set_dtstart (self->component, component_dt);
+ e_cal_component_commit_sequence (self->component);
+
+ g_object_notify (G_OBJECT (self), "date-start");
+
+ g_clear_pointer (&component_dt, e_cal_component_free_datetime);
+ }
+}
+
+/**
+ * gcal_event_get_description:
+ * @self a #GcalEvent
+ *
+ * Retrieves the description of @self.
+ *
+ * Returns: (transfer none): the description of the event.
+ */
+const gchar*
+gcal_event_get_description (GcalEvent *self)
+{
+ g_return_val_if_fail (GCAL_IS_EVENT (self), NULL);
+
+ return self->description ? self->description : "";
+}
+
+/**
+ * gcal_event_set_description:
+ * @self: a #GcalEvent
+ * @description: (nullable): the new description of the event
+ *
+ * Sets the description of the event.
+ */
+void
+gcal_event_set_description (GcalEvent *self,
+ const gchar *description)
+{
+ g_return_if_fail (GCAL_IS_EVENT (self));
+
+ if (g_strcmp0 (self->description, description) != 0)
+ {
+ ECalComponentText text_component;
+ GSList list;
+
+ self->description = g_strdup (description);
+
+ text_component.value = description;
+ text_component.altrep = NULL;
+
+ list.data = &text_component;
+ list.next = NULL;
+
+ e_cal_component_set_description_list (self->component, &list);
+ e_cal_component_commit_sequence (self->component);
+
+ g_object_notify (G_OBJECT (self), "description");
+ }
+}
+
+/**
+ * gcal_event_get_location:
+ * @self: a #GcalEvent
+ *
+ * Retrieves the location of the event.
+ *
+ * Returns: (transfer none): the location of the event
+ */
+const gchar*
+gcal_event_get_location (GcalEvent *self)
+{
+ const gchar *location;
+
+ g_return_val_if_fail (GCAL_IS_EVENT (self), NULL);
+
+ e_cal_component_get_location (self->component, &location);
+
+ return location ? location : "";
+}
+
+/**
+ * gcal_event_set_location:
+ * @self: a #GcalEvent
+ * @location: (nullable): the new location of the event
+ *
+ * Sets the location of the event.
+ */
+void
+gcal_event_set_location (GcalEvent *self,
+ const gchar *location)
+{
+ const gchar *current_location;
+
+ g_return_if_fail (GCAL_IS_EVENT (self));
+
+ current_location = gcal_event_get_location (self);
+
+ if (g_strcmp0 (current_location, location) != 0)
+ {
+ e_cal_component_set_location (self->component, location);
+
+ g_object_notify (G_OBJECT (self), "location");
+ }
+}
+
+/**
+ * gcal_event_get_source:
+ * @self: a #GcalEvent
+ *
+ * Retrieves the source of the event.
+ *
+ * Returns: (nullable): an #ESource.
+ */
+ESource*
+gcal_event_get_source (GcalEvent *self)
+{
+ g_return_val_if_fail (GCAL_IS_EVENT (self), NULL);
+
+ return self->source;
+}
+
+/**
+ * gcal_event_set_source:
+ * @self: a #GcalEvent
+ * @source: an #ESource
+ *
+ * Sets the source of this event. The color of the
+ * event is automatically tied with the source's
+ * color.
+ *
+ * The source should only be set once.
+ */
+void
+gcal_event_set_source (GcalEvent *self,
+ ESource *source)
+{
+ g_return_if_fail (GCAL_IS_EVENT (self));
+
+ if (self->source != source)
+ {
+ /* Remove previous binding */
+ g_clear_pointer (&self->color_binding, g_binding_unbind);
+
+ g_set_object (&self->source, source);
+
+ if (source)
+ {
+ ESourceSelectable *extension;
+ GdkRGBA color;
+
+ extension = E_SOURCE_SELECTABLE (e_source_get_extension (source, E_SOURCE_EXTENSION_CALENDAR));
+
+ /* calendar default color */
+ if (!gdk_rgba_parse (&color, e_source_selectable_get_color (extension)))
+ gdk_rgba_parse (&color, "#ffffff");
+
+ gcal_event_set_color (self, &color);
+
+ /* Bind the source's color with this event's color */
+ self->color_binding = g_object_bind_property_full (extension, "color",
+ self, "color",
+ G_BINDING_DEFAULT,
+ string_to_color,
+ NULL,
+ self,
+ NULL);
+ }
+
+ g_object_notify (G_OBJECT (self), "source");
+
+ gcal_event_update_uid_internal (self);
+ }
+}
+
+/**
+ * gcal_event_get_summary:
+ * @self: a #GcalEvent
+ *
+ * Retrieves the summary of this event.
+ *
+ * Returns: (transfer none): the summary of the event.
+ */
+const gchar*
+gcal_event_get_summary (GcalEvent *self)
+{
+ ECalComponentText summary;
+
+ g_return_val_if_fail (GCAL_IS_EVENT (self), NULL);
+
+ e_cal_component_get_summary (self->component, &summary);
+
+ return summary.value ? summary.value : "";
+}
+
+/**
+ * gcal_event_set_summary:
+ * @event: a #GcalEvent
+ * @summary: (nullable): the new summary of @event.
+ *
+ * Sets the summary of @event.
+ */
+void
+gcal_event_set_summary (GcalEvent *self,
+ const gchar *summary)
+{
+ const gchar *current_summary;
+
+ g_return_if_fail (GCAL_IS_EVENT (self));
+
+ current_summary = gcal_event_get_summary (self);
+
+ if (g_strcmp0 (current_summary, summary) != 0)
+ {
+ ECalComponentText text_component;
+
+ text_component.value = summary ? summary : "";
+ text_component.altrep = NULL;
+
+ e_cal_component_set_summary (self->component, &text_component);
+ e_cal_component_commit_sequence (self->component);
+
+ g_object_notify (G_OBJECT (self), "summary");
+ }
+}
+
+/**
+ * gcal_event_get_timezone:
+ * @self: a #GcalEvent
+ *
+ * Retrieves the event's timezone.
+ *
+ * Returns: (transfer none): a #GTimeZone
+ */
+GTimeZone*
+gcal_event_get_timezone (GcalEvent *self)
+{
+ g_return_val_if_fail (GCAL_IS_EVENT (self), NULL);
+
+ return self->timezone;
+}
+
+/**
+ * gcal_event_set_timezone:
+ * @self: a #GcalEvent
+ * @timezone: a #GTimeZone
+ *
+ * Sets the timezone of the event to @timezone.
+ */
+void
+gcal_event_set_timezone (GcalEvent *self,
+ GTimeZone *timezone)
+{
+ g_return_if_fail (GCAL_IS_EVENT (self));
+
+ if (self->timezone != timezone)
+ {
+ GDateTime *new_dtstart;
+
+ g_clear_pointer (&self->timezone, g_time_zone_unref);
+ self->timezone = g_time_zone_ref (timezone);
+
+ /* Swap the timezone from the component start & end dates*/
+ new_dtstart = g_date_time_to_timezone (self->dt_start, timezone);
+ gcal_event_set_date_start (self, new_dtstart);
+
+ if (self->dt_end)
+ {
+ GDateTime *new_dtend;
+
+ new_dtend = g_date_time_to_timezone (self->dt_end, timezone);
+ gcal_event_set_date_end (self, new_dtend);
+
+ g_clear_pointer (&new_dtstart, g_date_time_unref);
+ }
+
+ g_object_notify (G_OBJECT (self), "timezone");
+
+ g_clear_pointer (&new_dtstart, g_date_time_unref);
+ }
+}
+
+/**
+ * gcal_event_get_uid:
+ * @self: a #GcalEvent
+ *
+ * Retrieves the unique identifier of the event. The UID
+ * consists of '<source id>:<event id>:<recurrency id>'.
+ * If there's no recurrency assigned, the last field is
+ * ignored.
+ *
+ * Returns: (transfer none): the unique identifier of the event
+ */
+const gchar*
+gcal_event_get_uid (GcalEvent *self)
+{
+ g_return_val_if_fail (GCAL_IS_EVENT (self), NULL);
+
+ return self->uid;
+}
+
+/**
+ * gcal_event_is_multiday:
+ * @self: a #GcalEvent
+ *
+ * Whether the event spans more than 1 day.
+ *
+ * Returns: %TRUE if @self spans more than 1 day, %FALSE otherwise.
+ */
+gboolean
+gcal_event_is_multiday (GcalEvent *self)
+{
+ GDateTime *end_date;
+ gint n_days;
+
+ g_return_val_if_fail (GCAL_IS_EVENT (self), FALSE);
+
+ end_date = gcal_event_get_date_end (self);
+ n_days = g_date_time_difference (self->dt_start, end_date) / G_TIME_SPAN_DAY;
+
+ /*
+ * An all-day event with only 1 day of time span is treated as a
+ * single-day event.
+ */
+ if (self->all_day && n_days == 1)
+ return FALSE;
+
+ /*
+ * If any of the following fields are different, we're certain that
+ * it's a multiday event. Otherwise, we're certain it's NOT a multiday
+ * event.
+ */
+ if (g_date_time_get_year (self->dt_start) != g_date_time_get_year (end_date) ||
+ g_date_time_get_month (self->dt_start) != g_date_time_get_month (end_date) ||
+ g_date_time_get_day_of_month (self->dt_start) != g_date_time_get_day_of_month (end_date))
+ {
+ return TRUE;
+ }
+
+ return FALSE;
+}
+
+/**
+ * gcal_event_compare:
+ * @event1: a #GcalEvent
+ * @event2: a #GcalEvent
+ *
+ * Compare @event1 and @event2. It compares the start dates of
+ * the events and, when they have the same date (time is ignored),
+ * the @GcalEvent::all-day is the tiebreaker criteria.
+ *
+ * Events with equal start dates are sorted by the length.
+ *
+ * Returns: 1, 0 or -1 if @event1 is, respectively, lower, equal or
+ * greater than @event2.
+ */
+gint
+gcal_event_compare (GcalEvent *event1,
+ GcalEvent *event2)
+{
+ if (!event1 && !event2)
+ return 0;
+ if (!event1)
+ return 1;
+ else if (!event2)
+ return -1;
+
+ /*
+ * When the start date is the same, the priority is:
+ * 1. All day events (with largest time range)
+ * 2. All day events (with shortest time range)
+ * 3. Timed events (with largest time range)
+ * 4. Timed events (with shortest time range)
+ */
+ if (g_date_time_get_year (event1->dt_start) == g_date_time_get_year (event2->dt_start) &&
+ g_date_time_get_month (event1->dt_start) == g_date_time_get_month (event2->dt_start) &&
+ g_date_time_get_day_of_month (event1->dt_start) == g_date_time_get_day_of_month (event2->dt_start))
+ {
+ GTimeSpan span1, span2;
+
+ span1 = g_date_time_difference (event1->dt_start, gcal_event_get_date_end (event1));
+ span2 = g_date_time_difference (event2->dt_start, gcal_event_get_date_end (event2));
+
+ if (event1->all_day && event2->all_day)
+ {
+ // Case 1 & 2
+ return span1 - span2;
+ }
+ else if (event1->all_day)
+ {
+ return 1;
+ }
+ else if (event2->all_day)
+ {
+ return -1;
+ }
+ else
+ {
+ // Case 3 & 4
+ gint result;
+
+ result = g_date_time_compare (event1->dt_start, event2->dt_start);
+
+ if (result == 0)
+ return span1 - span2;
+ else
+ return result;
+ }
+ }
+
+ return g_date_time_compare (event1->dt_start, event2->dt_start);
+}
+
+/**
+ * gcal_event_compare_with_current:
+ * @event1: a #GcalEvent
+ * @event2: a #GcalEvent
+ * @current_time: the current time
+ *
+ * Compares @event1 and @event2 related to @current_time; the closest
+ * an event is from @current_time, the more priority it has.
+ *
+ * Returns: -1, 0 and 1 if @event1 is, respectively, farther, equal or
+ * closer from @current_time relative to @event2.
+ */
+gint
+gcal_event_compare_with_current (GcalEvent *event1,
+ GcalEvent *event2,
+ time_t *current_time)
+{
+ time_t time1, time2;
+ time_t diff1, diff2;
+
+ if (!event1 && !event2)
+ return 0;
+ if (!event1)
+ return 1;
+ else if (!event2)
+ return -1;
+
+ time1 = g_date_time_to_unix (event1->dt_start);
+ time2 = g_date_time_to_unix (event2->dt_start);
+ diff1 = time1 - *current_time;
+ diff2 = time2 - *current_time;
+
+ if (diff1 != diff2)
+ {
+ if (diff1 == 0)
+ return -1;
+ else if (diff2 == 0)
+ return 1;
+
+ if (diff1 > 0 && diff2 < 0)
+ return -1;
+ else if (diff2 > 0 && diff1 < 0)
+ return 1;
+ else if (diff1 < 0 && diff2 < 0)
+ return ABS (diff1) - ABS (diff2);
+ else if (diff1 > 0 && diff2 > 0)
+ return diff1 - diff2;
+ }
+
+ return 0;
+}
diff --git a/src/gcal-event.h b/src/gcal-event.h
new file mode 100644
index 0000000..e5bb0cd
--- /dev/null
+++ b/src/gcal-event.h
@@ -0,0 +1,98 @@
+/* gcal-event.h
+ *
+ * Copyright (C) 2016 Georges Basile Stavracas Neto <georges stavracas gmail com>
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * 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 General Public License
+ * along with this program. If not, see <http://www.gnu.org/licenses/>.
+ */
+
+#ifndef GCAL_EVENT_H
+#define GCAL_EVENT_H
+
+#include <glib-object.h>
+#include <gtk/gtk.h>
+#include <libecal/libecal.h>
+#include <libedataserver/libedataserver.h>
+
+G_BEGIN_DECLS
+
+#define GCAL_TYPE_EVENT (gcal_event_get_type())
+
+G_DECLARE_FINAL_TYPE (GcalEvent, gcal_event, GCAL, EVENT, GObject)
+
+GcalEvent* gcal_event_new (ESource *source,
+ ECalComponent *component);
+
+gboolean gcal_event_get_all_day (GcalEvent *self);
+
+void gcal_event_set_all_day (GcalEvent *self,
+ gboolean all_day);
+
+GdkRGBA* gcal_event_get_color (GcalEvent *self);
+
+void gcal_event_set_color (GcalEvent *self,
+ GdkRGBA *color);
+
+ECalComponent* gcal_event_get_component (GcalEvent *self);
+
+GDateTime* gcal_event_get_date_end (GcalEvent *self);
+
+void gcal_event_set_date_end (GcalEvent *self,
+ GDateTime *dt);
+
+GDateTime* gcal_event_get_date_start (GcalEvent *self);
+
+void gcal_event_set_date_start (GcalEvent *self,
+ GDateTime *dt);
+
+const gchar* gcal_event_get_description (GcalEvent *self);
+
+void gcal_event_set_description (GcalEvent *self,
+ const gchar *description);
+
+const gchar* gcal_event_get_location (GcalEvent *self);
+
+void gcal_event_set_location (GcalEvent *self,
+ const gchar *location);
+
+ESource* gcal_event_get_source (GcalEvent *self);
+
+void gcal_event_set_source (GcalEvent *self,
+ ESource *source);
+
+const gchar* gcal_event_get_summary (GcalEvent *self);
+
+void gcal_event_set_summary (GcalEvent *self,
+ const gchar *summary);
+
+GTimeZone* gcal_event_get_timezone (GcalEvent *self);
+
+void gcal_event_set_timezone (GcalEvent *self,
+ GTimeZone *timezone);
+
+const gchar* gcal_event_get_uid (GcalEvent *self);
+
+/* Utilities */
+
+gboolean gcal_event_is_multiday (GcalEvent *self);
+
+gint gcal_event_compare (GcalEvent *event1,
+ GcalEvent *event2);
+
+gint gcal_event_compare_with_current (GcalEvent *event1,
+ GcalEvent *event2,
+ time_t *current_time);
+
+G_END_DECLS
+
+#endif /* GCAL_EVENT_H */
diff --git a/src/gcal-utils.c b/src/gcal-utils.c
index 7969f28..6766eed 100644
--- a/src/gcal-utils.c
+++ b/src/gcal-utils.c
@@ -64,6 +64,53 @@ month_item[12] =
G_DEFINE_BOXED_TYPE (icaltimetype, icaltime, gcal_dup_icaltime, g_free)
icaltimetype*
+datetime_to_icaltime (GDateTime *dt)
+{
+ icaltimetype *idt;
+
+ if (!dt)
+ return NULL;
+
+ idt = g_new0 (icaltimetype, 1);
+
+ idt->year = g_date_time_get_year (dt);
+ idt->month = g_date_time_get_month (dt);
+ idt->day = g_date_time_get_day_of_month (dt);
+ idt->hour = g_date_time_get_hour (dt);
+ idt->minute = g_date_time_get_minute (dt);
+ idt->second = g_date_time_get_seconds (dt);
+ idt->is_date = (idt->hour == 0 &&
+ idt->minute == 0 &&
+ idt->second == 0);
+
+ return idt;
+}
+
+GDateTime*
+icaltime_to_datetime (const icaltimetype *date,
+ GTimeZone **timezone)
+{
+ GDateTime *dt;
+ GTimeZone *tz;
+
+ tz = date->zone ? g_time_zone_new (icaltime_get_tzid (*date)) : g_time_zone_new_utc ();
+ dt = g_date_time_new (tz,
+ date->year,
+ date->month,
+ date->day,
+ date->hour,
+ date->minute,
+ date->second);
+
+ if (timezone)
+ *timezone = tz;
+ else
+ g_clear_pointer (&tz, g_time_zone_unref);
+
+ return dt;
+}
+
+icaltimetype*
gcal_dup_icaltime (const icaltimetype *date)
{
icaltimetype *new_date;
diff --git a/src/gcal-utils.h b/src/gcal-utils.h
index cbc8e7d..4d009df 100644
--- a/src/gcal-utils.h
+++ b/src/gcal-utils.h
@@ -52,6 +52,11 @@ const gchar* (*GcalTranslateFunc) (GtkWidget
GType icaltime_get_type (void) G_GNUC_CONST;
+icaltimetype* datetime_to_icaltime (GDateTime *dt);
+
+GDateTime* icaltime_to_datetime (const icaltimetype *date,
+ GTimeZone **timezone);
+
icaltimetype* gcal_dup_icaltime (const icaltimetype *date);
gchar* gcal_get_weekday (gint i);
[
Date Prev][
Date Next] [
Thread Prev][
Thread Next]
[
Thread Index]
[
Date Index]
[
Author Index]