[gnome-calendar/wip/gbsneto/date-chooser: 3/5] date-selector: turn into a entry



commit 5ea9edff74855a86341c776afb8672b27412f7ca
Author: Georges Basile Stavracas Neto <georges stavracas gmail com>
Date:   Fri Jul 22 01:03:23 2016 -0300

    date-selector: turn into a entry
    
    After making the date selector use the brand
    new date chooser widget, it is now time to make
    it an entry.
    
    This commit, then turns GcalDateSelector into a
    GtkEntry subclass, and add all the plumbing that
    is necessary to make it work as such.
    
    As a side effect, this commit adds rudimentary
    natural language support for parsing dates.

 data/ui/date-selector.ui |   14 +++----
 src/gcal-date-selector.c |   99 ++++++++++++++++++++++++++++++++++++++++------
 src/gcal-date-selector.h |    2 +-
 3 files changed, 93 insertions(+), 22 deletions(-)
---
diff --git a/data/ui/date-selector.ui b/data/ui/date-selector.ui
index 3375c8a..6707289 100644
--- a/data/ui/date-selector.ui
+++ b/data/ui/date-selector.ui
@@ -1,14 +1,12 @@
 <?xml version="1.0" encoding="UTF-8"?>
 <interface>
   <requires lib="gtk+" version="3.12"/>
-  <template class="GcalDateSelector" parent="GtkMenuButton">
-    <property name="use_popover">True</property>
-    <property name="popover">date_selector_popover</property>
-    <child>
-      <object class="GtkLabel" id="date_label">
-        <property name="visible">True</property>
-      </object>
-    </child>
+  <template class="GcalDateSelector" parent="GtkEntry">
+    <property name="visible">True</property>
+    <property name="width-chars">16</property>
+    <property name="max-width-chars">16</property>
+    <property name="secondary-icon-name">pan-down-symbolic</property>
+    <signal name="icon-press" handler="icon_pressed_cb" object="GcalDateSelector" swapped="no"/>
   </template>
   <object class="GtkPopover" id="date_selector_popover">
     <property name="position">bottom</property>
diff --git a/src/gcal-date-selector.c b/src/gcal-date-selector.c
index 1769214..56e8ba4 100644
--- a/src/gcal-date-selector.c
+++ b/src/gcal-date-selector.c
@@ -27,11 +27,11 @@
 
 struct _GcalDateSelector
 {
-  GtkMenuButton parent;
+  GtkEntry     parent;
 
   /* widgets */
   GtkWidget   *date_chooser;
-  GtkWidget   *date_label;
+  GtkWidget   *date_selector_popover;
 };
 
 enum
@@ -41,10 +41,10 @@ enum
   LAST_PROP
 };
 
-G_DEFINE_TYPE (GcalDateSelector, gcal_date_selector, GTK_TYPE_MENU_BUTTON);
+G_DEFINE_TYPE (GcalDateSelector, gcal_date_selector, GTK_TYPE_ENTRY);
 
 static void
-update_label (GcalDateSelector *self)
+update_text (GcalDateSelector *self)
 {
   GDateTime *date;
   gchar *label;
@@ -54,19 +54,62 @@ update_label (GcalDateSelector *self)
   /* rebuild the date label */
   label = g_date_time_format (date, "%x");
 
-  gtk_label_set_label (GTK_LABEL (self->date_label), label);
+  gtk_entry_set_text (GTK_ENTRY (self), label);
   g_free (label);
 }
 
 static void
 calendar_day_selected (GcalDateSelector *self)
 {
-  update_label (self);
+  update_text (self);
 
   g_object_notify (G_OBJECT (self), "date");
 }
 
 static void
+parse_date (GcalDateSelector *self)
+{
+  GDateTime *new_date;
+  GDate parsed_date;
+
+  g_date_clear (&parsed_date, 1);
+  g_date_set_parse (&parsed_date, gtk_entry_get_text (GTK_ENTRY (self)));
+
+  if (!g_date_valid (&parsed_date))
+    {
+      update_text (self);
+      return;
+    }
+
+  new_date = g_date_time_new_local (g_date_get_year (&parsed_date),
+                                    g_date_get_month (&parsed_date),
+                                    g_date_get_day (&parsed_date),
+                                    0, 0, 0);
+
+  gcal_date_selector_set_date (self, new_date);
+
+  g_clear_pointer (&new_date, g_date_time_unref);
+}
+
+static void
+icon_pressed_cb (GcalDateSelector     *self,
+                 GtkEntryIconPosition  position,
+                 GdkEvent             *event)
+{
+  GdkRectangle icon_bounds;
+
+  gtk_entry_get_icon_area (GTK_ENTRY (self), position, &icon_bounds);
+
+  /* HACK: seems like the popover is 2.5 px misplaced */
+  icon_bounds.x += 3;
+
+  gtk_popover_set_relative_to (GTK_POPOVER (self->date_selector_popover), GTK_WIDGET (self));
+  gtk_popover_set_pointing_to (GTK_POPOVER (self->date_selector_popover), &icon_bounds);
+
+  gtk_widget_show (self->date_selector_popover);
+}
+
+static void
 gcal_date_selector_get_property (GObject    *object,
                                  guint       prop_id,
                                  GValue     *value,
@@ -104,15 +147,45 @@ gcal_date_selector_set_property (GObject      *object,
     }
 }
 
+static gboolean
+gcal_date_selector_focus_in_event (GtkWidget     *widget,
+                                   GdkEventFocus *event)
+{
+  parse_date (GCAL_DATE_SELECTOR (widget));
+
+  return GTK_WIDGET_CLASS (gcal_date_selector_parent_class)->focus_in_event (widget, event);
+}
+
+static gboolean
+gcal_date_selector_focus_out_event (GtkWidget     *widget,
+                                    GdkEventFocus *event)
+{
+  parse_date (GCAL_DATE_SELECTOR (widget));
+
+  return GTK_WIDGET_CLASS (gcal_date_selector_parent_class)->focus_out_event (widget, event);
+}
+
+static void
+gcal_date_selector_activate (GtkEntry *entry)
+{
+  parse_date (GCAL_DATE_SELECTOR (entry));
+}
+
 static void
 gcal_date_selector_class_init (GcalDateSelectorClass *klass)
 {
-  GObjectClass *object_class;
+  GtkWidgetClass *widget_class = GTK_WIDGET_CLASS (klass);
+  GtkEntryClass *entry_class = GTK_ENTRY_CLASS (klass);
+  GObjectClass *object_class = G_OBJECT_CLASS (klass);
 
-  object_class = G_OBJECT_CLASS (klass);
   object_class->get_property = gcal_date_selector_get_property;
   object_class->set_property = gcal_date_selector_set_property;
 
+  widget_class->focus_in_event = gcal_date_selector_focus_in_event;
+  widget_class->focus_out_event = gcal_date_selector_focus_out_event;
+
+  entry_class->activate = gcal_date_selector_activate;
+
   /**
    * GcalDateSelector::date:
    *
@@ -126,18 +199,18 @@ gcal_date_selector_class_init (GcalDateSelectorClass *klass)
                                                        G_TYPE_DATE_TIME,
                                                        G_PARAM_READWRITE));
 
-  gtk_widget_class_set_template_from_resource (GTK_WIDGET_CLASS (klass), 
"/org/gnome/calendar/date-selector.ui");
+  gtk_widget_class_set_template_from_resource (widget_class, "/org/gnome/calendar/date-selector.ui");
 
-  gtk_widget_class_bind_template_child (GTK_WIDGET_CLASS (klass), GcalDateSelector, date_chooser);
-  gtk_widget_class_bind_template_child (GTK_WIDGET_CLASS (klass), GcalDateSelector, date_label);
+  gtk_widget_class_bind_template_child (widget_class, GcalDateSelector, date_chooser);
+  gtk_widget_class_bind_template_child (widget_class, GcalDateSelector, date_selector_popover);
 
-  gtk_widget_class_bind_template_callback (GTK_WIDGET_CLASS (klass), calendar_day_selected);
+  gtk_widget_class_bind_template_callback (widget_class, calendar_day_selected);
+  gtk_widget_class_bind_template_callback (widget_class, icon_pressed_cb);
 }
 
 static void
 gcal_date_selector_init (GcalDateSelector *self)
 {
-
   gtk_widget_set_has_window (GTK_WIDGET (self), FALSE);
 
   gtk_widget_init_template (GTK_WIDGET (self));
diff --git a/src/gcal-date-selector.h b/src/gcal-date-selector.h
index 326d2a5..1d20c52 100644
--- a/src/gcal-date-selector.h
+++ b/src/gcal-date-selector.h
@@ -26,7 +26,7 @@ G_BEGIN_DECLS
 
 #define GCAL_TYPE_DATE_SELECTOR             (gcal_date_selector_get_type ())
 
-G_DECLARE_FINAL_TYPE (GcalDateSelector, gcal_date_selector, GCAL, DATE_SELECTOR, GtkMenuButton)
+G_DECLARE_FINAL_TYPE (GcalDateSelector, gcal_date_selector, GCAL, DATE_SELECTOR, GtkEntry)
 
 GtkWidget*       gcal_date_selector_new             (void);
 


[Date Prev][Date Next]   [Thread Prev][Thread Next]   [Thread Index] [Date Index] [Author Index]