]> Pileus Git - ~andy/gtk/blobdiff - tests/testcalendar.c
stylecontext: Do invalidation on first resize container
[~andy/gtk] / tests / testcalendar.c
index 0ab4409eed0ce4bb522e020a35553497c18c0aea..84de5ec139c15b877250439f7d024687c5c76823 100644 (file)
@@ -3,43 +3,49 @@
  * Copyright (C) 1998 Cesar Miquel, Shawn T. Amundson, Mattias Grönlund
  * Copyright (C) 2000 Tony Gale
  *
- * 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 2 of the License, or
+ * This library 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; either version 2.1 of the License, or
  * (at your option) any later version.
  *
- * This program is distributed in the hope that it will be useful,
+ * This library 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.
+ * GNU Lesser General Public License for more details.
  *
- * You should have received a copy of the GNU General Public License
- * along with this program; if not, write to the Free Software
- * Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.
+ * 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/>.
  */
 
-#include <config.h>
+#include "config.h"
 #include <stdio.h>
 #include <string.h>
 #include <gtk/gtk.h>
 
-#define DEF_PAD 10
-#define DEF_PAD_SMALL 5
+#define DEF_PAD 12
+#define DEF_PAD_SMALL 6
 
 #define TM_YEAR_BASE 1900
 
-typedef struct _CalendarData {
-  GtkWidget *flag_checkboxes[5];
-  gboolean  settings[5];
+typedef struct _CalendarData
+{
+  GtkWidget *calendar_widget;
+  GtkWidget *flag_checkboxes[6];
+  gboolean  settings[6];
   GtkWidget *font_dialog;
   GtkWidget *window;
   GtkWidget *prev2_sig;
   GtkWidget *prev_sig;
   GtkWidget *last_sig;
   GtkWidget *month;
+
+  GHashTable    *details_table;
+  GtkTextBuffer *details_buffer;
+  gulong         details_changed;
 } CalendarData;
 
-enum {
+enum
+{
   calendar_show_header,
   calendar_show_days,
   calendar_month_change, 
@@ -51,7 +57,8 @@ enum {
  * GtkCalendar
  */
 
-void calendar_date_to_string (CalendarData *data,
+static void
+calendar_date_to_string (CalendarData *data,
                              char         *buffer,
                              gint          buff_len)
 {
@@ -66,7 +73,51 @@ void calendar_date_to_string (CalendarData *data,
   g_date_free (date);
 }
 
-void calendar_set_signal_strings (char         *sig_str,
+static void
+calendar_set_detail (CalendarData *data,
+                     guint         year,
+                     guint         month,
+                     guint         day,
+                     gchar        *detail)
+{
+  gchar *key = g_strdup_printf ("%04d-%02d-%02d", year, month + 1, day);
+  g_hash_table_replace (data->details_table, key, detail);
+}
+
+static gchar*
+calendar_get_detail (CalendarData *data,
+                     guint         year,
+                     guint         month,
+                     guint         day)
+{
+  const gchar *detail;
+  gchar *key;
+
+  key = g_strdup_printf ("%04d-%02d-%02d", year, month + 1, day);
+  detail = g_hash_table_lookup (data->details_table, key);
+  g_free (key);
+
+  return (detail ? g_strdup (detail) : NULL);
+}
+
+static void
+calendar_update_details (CalendarData *data)
+{
+  guint year, month, day;
+  gchar *detail;
+
+  gtk_calendar_get_date (GTK_CALENDAR (data->calendar_widget), &year, &month, &day);
+  detail = calendar_get_detail (data, year, month, day);
+
+  g_signal_handler_block (data->details_buffer, data->details_changed);
+  gtk_text_buffer_set_text (data->details_buffer, detail ? detail : "", -1);
+  g_signal_handler_unblock (data->details_buffer, data->details_changed);
+
+  g_free (detail);
+}
+
+static void
+calendar_set_signal_strings (char         *sig_str,
                                  CalendarData *data)
 {
   const gchar *prev_sig;
@@ -79,7 +130,8 @@ void calendar_set_signal_strings (char         *sig_str,
   gtk_label_set_text (GTK_LABEL (data->last_sig), sig_str);
 }
 
-void calendar_month_changed (GtkWidget    *widget,
+static void
+calendar_month_changed (GtkWidget    *widget,
                              CalendarData *data)
 {
   char buffer[256] = "month_changed: ";
@@ -88,16 +140,20 @@ void calendar_month_changed (GtkWidget    *widget,
   calendar_set_signal_strings (buffer, data);
 }
 
-void calendar_day_selected (GtkWidget    *widget,
+static void
+calendar_day_selected (GtkWidget    *widget,
                             CalendarData *data)
 {
   char buffer[256] = "day_selected: ";
 
   calendar_date_to_string (data, buffer+14, 256-14);
   calendar_set_signal_strings (buffer, data);
+
+  calendar_update_details (data);
 }
 
-void calendar_day_selected_double_click (GtkWidget    *widget,
+static void
+calendar_day_selected_double_click (GtkWidget    *widget,
                                          CalendarData *data)
 {
   char buffer[256] = "day_selected_double_click: ";
@@ -105,18 +161,17 @@ void calendar_day_selected_double_click (GtkWidget    *widget,
 
   calendar_date_to_string (data, buffer+27, 256-27);
   calendar_set_signal_strings (buffer, data);
-
   gtk_calendar_get_date (GTK_CALENDAR (data->window),
-                        NULL, NULL, &day);
+                         NULL, NULL, &day);
 
-  if (GTK_CALENDAR (data->window)->marked_date[day-1] == 0) {
-    gtk_calendar_mark_day (GTK_CALENDAR (data->window), day);
-  } else { 
+  if (gtk_calendar_get_day_is_marked (GTK_CALENDAR (data->window), day))
     gtk_calendar_unmark_day (GTK_CALENDAR (data->window), day);
-  }
+  else
+    gtk_calendar_mark_day (GTK_CALENDAR (data->window), day);
 }
 
-void calendar_prev_month (GtkWidget    *widget,
+static void
+calendar_prev_month (GtkWidget    *widget,
                           CalendarData *data)
 {
   char buffer[256] = "prev_month: ";
@@ -125,8 +180,9 @@ void calendar_prev_month (GtkWidget    *widget,
   calendar_set_signal_strings (buffer, data);
 }
 
-void calendar_next_month( GtkWidget    *widget,
-                            CalendarData *data )
+static void
+calendar_next_month (GtkWidget    *widget,
+                     CalendarData *data)
 {
   char buffer[256] = "next_month: ";
 
@@ -134,8 +190,9 @@ void calendar_next_month( GtkWidget    *widget,
   calendar_set_signal_strings (buffer, data);
 }
 
-void calendar_prev_year( GtkWidget    *widget,
-                            CalendarData *data )
+static void
+calendar_prev_year (GtkWidget    *widget,
+                    CalendarData *data)
 {
   char buffer[256] = "prev_year: ";
 
@@ -143,8 +200,9 @@ void calendar_prev_year( GtkWidget    *widget,
   calendar_set_signal_strings (buffer, data);
 }
 
-void calendar_next_year( GtkWidget    *widget,
-                            CalendarData *data )
+static void
+calendar_next_year (GtkWidget    *widget,
+                    CalendarData *data)
 {
   char buffer[256] = "next_year: ";
 
@@ -153,155 +211,238 @@ void calendar_next_year( GtkWidget    *widget,
 }
 
 
-void calendar_set_flags( CalendarData *calendar )
+static void
+calendar_set_flags (CalendarData *calendar)
 {
-  gint i;
-  gint options=0;
-  for (i=0;i<5;i++) 
+  gint options = 0, i;
+
+  for (i = 0; i < G_N_ELEMENTS (calendar->settings); i++)
     if (calendar->settings[i])
-      {
-       options=options + (1<<i);
-      }
+      options=options + (1 << i);
+
   if (calendar->window)
     gtk_calendar_set_display_options (GTK_CALENDAR (calendar->window), options);
 }
 
-void calendar_toggle_flag( GtkWidget    *toggle,
-                           CalendarData *calendar )
+static void
+calendar_toggle_flag (GtkWidget    *toggle,
+                      CalendarData *calendar)
 {
   gint i;
-  gint j;
-  j=0;
-  for (i=0; i<5; i++)
+
+  for (i = 0; i < G_N_ELEMENTS (calendar->flag_checkboxes); i++)
     if (calendar->flag_checkboxes[i] == toggle)
-      j = i;
+      calendar->settings[i] = gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (toggle));
 
-  calendar->settings[j]=!calendar->settings[j];
   calendar_set_flags(calendar);
   
 }
 
-void calendar_font_selection_ok (GtkWidget    *button,
+void calendar_select_font (GtkWidget    *button,
                                  CalendarData *calendar)
 {
-  GtkRcStyle *style;
-  char *font_name;
+  const char *font = NULL;
+  PangoFontDescription *font_desc;
 
   if (calendar->window)
     {
-      font_name = gtk_font_selection_dialog_get_font_name (GTK_FONT_SELECTION_DIALOG(calendar->font_dialog));
-      if (font_name) 
-       {
-         style = gtk_rc_style_new ();
-         pango_font_description_free (style->font_desc);
-         style->font_desc = pango_font_description_from_string (font_name);
-         gtk_widget_modify_style (calendar->window, style);
-         g_free (font_name);
-       }
+      font = gtk_font_button_get_font_name (GTK_FONT_BUTTON (button));
+      font_desc = pango_font_description_from_string (font);
+      gtk_widget_override_font (calendar->window, font_desc);
+      pango_font_description_free (font_desc);
     }
+}
 
-  gtk_widget_destroy (calendar->font_dialog);
+static gchar*
+calendar_detail_cb (GtkCalendar *calendar,
+                    guint        year,
+                    guint        month,
+                    guint        day,
+                    gpointer     data)
+{
+  return calendar_get_detail (data, year, month, day);
 }
 
-void calendar_select_font( GtkWidget    *button,
-                           CalendarData *calendar )
+static void
+calendar_details_changed (GtkTextBuffer *buffer,
+                          CalendarData  *data)
 {
-  GtkWidget *window;
+  GtkTextIter start, end;
+  guint year, month, day;
+  gchar *detail;
 
-  if (!calendar->font_dialog) {
-    window = gtk_font_selection_dialog_new ("Font Selection Dialog");
-    g_return_if_fail(GTK_IS_FONT_SELECTION_DIALOG(window));
-    calendar->font_dialog = window;
-    
-    gtk_window_set_position (GTK_WINDOW (window), GTK_WIN_POS_MOUSE);
-    
-    g_signal_connect (window, "destroy",
-                     G_CALLBACK (gtk_widget_destroyed),
-                     &calendar->font_dialog);
-    
-    g_signal_connect (GTK_FONT_SELECTION_DIALOG (window)->ok_button,
-                     "clicked", G_CALLBACK (calendar_font_selection_ok),
-                     calendar);
-    g_signal_connect_swapped (GTK_FONT_SELECTION_DIALOG (window)->cancel_button,
-                            "clicked", G_CALLBACK (gtk_widget_destroy), 
-                            calendar->font_dialog);
-  }
-  window=calendar->font_dialog;
-  if (!GTK_WIDGET_VISIBLE (window))
-    gtk_widget_show (window);
+  gtk_text_buffer_get_start_iter(buffer, &start);
+  gtk_text_buffer_get_end_iter(buffer, &end);
+
+  gtk_calendar_get_date (GTK_CALENDAR (data->calendar_widget), &year, &month, &day);
+  detail = gtk_text_buffer_get_text (buffer, &start, &end, FALSE);
+
+  if (!detail[0])
+    {
+      g_free (detail);
+      detail = NULL;
+    }
+
+  calendar_set_detail (data, year, month, day, detail);
+  gtk_widget_queue_resize (data->calendar_widget);
+}
+
+static void
+demonstrate_details (CalendarData *data)
+{
+  static char *rainbow[] = { "#900", "#980", "#390", "#095", "#059", "#309", "#908" };
+  GtkCalendar *calendar = GTK_CALENDAR (data->calendar_widget);
+  guint year, month, day;
+  gchar *detail;
+
+  gtk_calendar_get_date (calendar,
+                         &year, &month, &day);
+
+  for (day = 0; day < 29; ++day)
+    {
+      detail = g_strdup_printf ("<span color='%s'>yadda\n"
+                                "(%04d-%02d-%02d)</span>",
+                                rainbow[(day - 1) % 7], year, month, day);
+      calendar_set_detail (data, year, month, day, detail);
+   }
+
+  gtk_widget_queue_resize (data->calendar_widget);
+  calendar_update_details (data);
+}
+
+static void
+reset_details (CalendarData *data)
+{
+  g_hash_table_remove_all (data->details_table);
+  gtk_widget_queue_resize (data->calendar_widget);
+  calendar_update_details (data);
+}
+
+static void
+calendar_toggle_details (GtkWidget    *widget,
+                         CalendarData *data)
+{
+  if (gtk_toggle_button_get_active (GTK_TOGGLE_BUTTON (widget)))
+    gtk_calendar_set_detail_func (GTK_CALENDAR (data->calendar_widget),
+                                  calendar_detail_cb, data, NULL);
   else
-    gtk_widget_destroy (window);
+    gtk_calendar_set_detail_func (GTK_CALENDAR (data->calendar_widget),
+                                  NULL, NULL, NULL);
+}
+
+static GtkWidget*
+create_expander (const char *caption,
+                 GtkWidget  *child,
+                 gdouble     xscale,
+                 gdouble     yscale)
+{
+  GtkWidget *expander = gtk_expander_new ("");
+  GtkWidget *label = gtk_expander_get_label_widget (GTK_EXPANDER (expander));
+  GtkWidget *align = gtk_alignment_new (0, 0, xscale, yscale);
+
+  gtk_alignment_set_padding (GTK_ALIGNMENT (align), 6, 0, 18, 0);
+  gtk_label_set_markup (GTK_LABEL (label), caption);
+
+  gtk_container_add (GTK_CONTAINER (expander), align);
+  gtk_container_add (GTK_CONTAINER (align), child);
 
+  return expander;
 }
 
-void create_calendar(void)
+static GtkWidget*
+create_frame (const char *caption,
+              GtkWidget  *child,
+              gdouble     xscale,
+              gdouble     yscale)
+{
+  GtkWidget *frame = gtk_frame_new ("");
+  GtkWidget *label = gtk_frame_get_label_widget (GTK_FRAME (frame));
+  GtkWidget *align = gtk_alignment_new (0, 0, xscale, yscale);
+
+  gtk_frame_set_shadow_type (GTK_FRAME (frame), GTK_SHADOW_NONE);
+  gtk_alignment_set_padding (GTK_ALIGNMENT (align), 6, 0, 18, 0);
+  gtk_label_set_markup (GTK_LABEL (label), caption);
+
+  gtk_container_add (GTK_CONTAINER (frame), align);
+  gtk_container_add (GTK_CONTAINER (align), child);
+
+  return frame;
+}
+
+static void
+detail_width_changed (GtkSpinButton *button,
+                      CalendarData  *data)
+{
+  gint value = (gint) gtk_spin_button_get_value (button);
+  gtk_calendar_set_detail_width_chars (GTK_CALENDAR (data->calendar_widget), value);
+}
+
+static void
+detail_height_changed (GtkSpinButton *button,
+                      CalendarData  *data)
+{
+  gint value = (gint) gtk_spin_button_get_value (button);
+  gtk_calendar_set_detail_height_rows (GTK_CALENDAR (data->calendar_widget), value);
+}
+
+static void
+create_calendar(void)
 {
-  GtkWidget *window;
-  GtkWidget *vbox, *vbox2, *vbox3;
-  GtkWidget *hbox;
-  GtkWidget *hbbox;
-  GtkWidget *calendar;
-  GtkWidget *toggle;
-  GtkWidget *button;
-  GtkWidget *frame;
-  GtkWidget *separator;
-  GtkWidget *label;
-  GtkWidget *bbox;
   static CalendarData calendar_data;
+
+  GtkWidget *window, *hpaned, *vbox, *rpane, *hbox;
+  GtkWidget *calendar, *toggle, *scroller, *button;
+  GtkWidget *frame, *label, *bbox, *align, *details;
+
+  GtkSizeGroup *size;
+  GtkStyleContext *context;
+  PangoFontDescription *font_desc;
+  gchar *font;
   gint i;
   
   struct {
+    gboolean init;
     char *label;
   } flags[] =
     {
-      { "Show Heading" },
-      { "Show Day Names" },
-      { "No Month Change" },
-      { "Show Week Numbers" },
-      { "Week Start Monday" }
+      { TRUE,  "Show _Heading" },
+      { TRUE,  "Show Day _Names" },
+      { FALSE, "No Month _Change" },
+      { TRUE,  "Show _Week Numbers" },
+      { FALSE, "Week Start _Monday" },
+      { TRUE,  "Show De_tails" },
     };
 
-  
   calendar_data.window = NULL;
   calendar_data.font_dialog = NULL;
+  calendar_data.details_table = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
 
-  for (i=0; i<5; i++) {
-    calendar_data.settings[i]=0;
-  }
+  for (i = 0; i < G_N_ELEMENTS (calendar_data.settings); i++)
+    calendar_data.settings[i] = 0;
 
   window = gtk_window_new (GTK_WINDOW_TOPLEVEL);
   gtk_window_set_title (GTK_WINDOW (window), "GtkCalendar Example");
-  gtk_container_set_border_width (GTK_CONTAINER (window), 5);
+  gtk_container_set_border_width (GTK_CONTAINER (window), 12);
   g_signal_connect (window, "destroy",
                    G_CALLBACK (gtk_main_quit),
                    NULL);
   g_signal_connect (window, "delete-event",
                    G_CALLBACK (gtk_false),
                    NULL);
-  gtk_window_set_resizable (GTK_WINDOW (window), FALSE);
 
-  vbox = gtk_vbox_new (FALSE, DEF_PAD);
-  gtk_container_add (GTK_CONTAINER (window), vbox);
+  hpaned = gtk_paned_new (GTK_ORIENTATION_HORIZONTAL);
 
-  /*
-   * The top part of the window, Calendar, flags and fontsel.
-   */
+  /* Calendar widget */
 
-  hbox = gtk_hbox_new(FALSE, DEF_PAD);
-  gtk_box_pack_start (GTK_BOX(vbox), hbox, TRUE, TRUE, DEF_PAD);
-  hbbox = gtk_hbutton_box_new();
-  gtk_box_pack_start(GTK_BOX(hbox), hbbox, FALSE, FALSE, DEF_PAD);
-  gtk_button_box_set_layout(GTK_BUTTON_BOX(hbbox), GTK_BUTTONBOX_SPREAD);
-  gtk_box_set_spacing(GTK_BOX(hbbox), 5);
+  calendar = gtk_calendar_new ();
+  calendar_data.calendar_widget = calendar;
+  frame = create_frame ("<b>Calendar</b>", calendar, 0, 0);
+  gtk_paned_pack1 (GTK_PANED (hpaned), frame, TRUE, FALSE);
 
-  /* Calendar widget */
-  frame = gtk_frame_new("Calendar");
-  gtk_box_pack_start(GTK_BOX(hbbox), frame, FALSE, TRUE, DEF_PAD);
-  calendar=gtk_calendar_new();
   calendar_data.window = calendar;
   calendar_set_flags(&calendar_data);
   gtk_calendar_mark_day (GTK_CALENDAR (calendar), 19); 
-  gtk_container_add (GTK_CONTAINER (frame), calendar);
+
   g_signal_connect (calendar, "month_changed", 
                    G_CALLBACK (calendar_month_changed),
                    &calendar_data);
@@ -324,82 +465,212 @@ void create_calendar(void)
                    G_CALLBACK (calendar_next_year),
                    &calendar_data);
 
+  rpane = gtk_box_new (GTK_ORIENTATION_VERTICAL, DEF_PAD_SMALL);
+  gtk_paned_pack2 (GTK_PANED (hpaned), rpane, FALSE, FALSE);
+
+  /* Build the right font-button */
+
+  vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, DEF_PAD_SMALL);
+  frame = create_frame ("<b>Options</b>", vbox, 1, 0);
+  gtk_box_pack_start (GTK_BOX (rpane), frame, FALSE, TRUE, 0);
+  size = gtk_size_group_new (GTK_SIZE_GROUP_HORIZONTAL);
+
+  context = gtk_widget_get_style_context (calendar);
+  gtk_style_context_get (context, GTK_STATE_FLAG_NORMAL, &font_desc, NULL);
+  font = pango_font_description_to_string (font_desc);
+  button = gtk_font_button_new_with_font (font);
+  g_free (font);
+  pango_font_description_free (font_desc);
+
+  g_signal_connect (button, "font-set",
+                    G_CALLBACK(calendar_select_font),
+                    &calendar_data);
+
+  label = gtk_label_new_with_mnemonic ("_Font:");
+  gtk_label_set_mnemonic_widget (GTK_LABEL (label), button);
+  gtk_widget_set_halign (label, GTK_ALIGN_START);
+  gtk_widget_set_valign (label, GTK_ALIGN_CENTER);
+  gtk_size_group_add_widget (size, label);
+
+  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, DEF_PAD_SMALL);
+  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, TRUE, 0);
+  gtk_box_pack_start (GTK_BOX (hbox), button, FALSE, TRUE, 0);
+  gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, TRUE, 0);
+
+  /* Build the width entry */
+
+  button = gtk_spin_button_new_with_range (0, 127, 1);
+  gtk_spin_button_set_value (GTK_SPIN_BUTTON (button),
+                             gtk_calendar_get_detail_width_chars (GTK_CALENDAR (calendar)));
+
+  g_signal_connect (button, "value-changed",
+                    G_CALLBACK (detail_width_changed),
+                    &calendar_data);
+
+  label = gtk_label_new_with_mnemonic ("Details W_idth:");
+  gtk_label_set_mnemonic_widget (GTK_LABEL (label), button);
+  gtk_widget_set_halign (label, GTK_ALIGN_START);
+  gtk_widget_set_valign (label, GTK_ALIGN_CENTER);
+  gtk_size_group_add_widget (size, label);
+
+  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, DEF_PAD_SMALL);
+  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, TRUE, 0);
+  gtk_box_pack_start (GTK_BOX (hbox), button, FALSE, TRUE, 0);
+  gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, TRUE, 0);
+
+  /* Build the height entry */
+
+  button = gtk_spin_button_new_with_range (0, 127, 1);
+  gtk_spin_button_set_value (GTK_SPIN_BUTTON (button),
+                             gtk_calendar_get_detail_height_rows (GTK_CALENDAR (calendar)));
+
+  g_signal_connect (button, "value-changed",
+                    G_CALLBACK (detail_height_changed),
+                    &calendar_data);
 
-  separator = gtk_vseparator_new ();
-  gtk_box_pack_start (GTK_BOX (hbox), separator, FALSE, TRUE, 0);
+  label = gtk_label_new_with_mnemonic ("Details H_eight:");
+  gtk_label_set_mnemonic_widget (GTK_LABEL (label), button);
+  gtk_widget_set_halign (label, GTK_ALIGN_START);
+  gtk_widget_set_valign (label, GTK_ALIGN_CENTER);
+  gtk_size_group_add_widget (size, label);
 
-  vbox2 = gtk_vbox_new(FALSE, DEF_PAD);
-  gtk_box_pack_start(GTK_BOX(hbox), vbox2, FALSE, FALSE, DEF_PAD);
+  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, DEF_PAD_SMALL);
+  gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, TRUE, 0);
+  gtk_box_pack_start (GTK_BOX (hbox), button, FALSE, TRUE, 0);
+  gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, TRUE, 0);
+
+  /* Build the right details frame */
+
+  vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, DEF_PAD_SMALL);
+  frame = create_frame ("<b>Details</b>", vbox, 1, 1);
+  gtk_box_pack_start (GTK_BOX (rpane), frame, FALSE, TRUE, 0);
+
+  details = gtk_text_view_new();
+  calendar_data.details_buffer = gtk_text_view_get_buffer (GTK_TEXT_VIEW (details));
+
+  calendar_data.details_changed = g_signal_connect (calendar_data.details_buffer, "changed",
+                                                    G_CALLBACK (calendar_details_changed),
+                                                    &calendar_data);
+
+  scroller = gtk_scrolled_window_new (NULL, NULL);
+  gtk_container_add (GTK_CONTAINER (scroller), details);
+
+  gtk_scrolled_window_set_shadow_type (GTK_SCROLLED_WINDOW (scroller),
+                                       GTK_SHADOW_IN);
+  gtk_scrolled_window_set_policy (GTK_SCROLLED_WINDOW (scroller),
+                                  GTK_POLICY_AUTOMATIC,
+                                  GTK_POLICY_AUTOMATIC);
+
+  gtk_box_pack_start (GTK_BOX (vbox), scroller, FALSE, TRUE, 0);
+
+  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, DEF_PAD_SMALL);
+  align = gtk_alignment_new (0.0, 0.5, 0.0, 0.0);
+  gtk_container_add (GTK_CONTAINER (align), hbox);
+  gtk_box_pack_start (GTK_BOX (vbox), align, FALSE, TRUE, 0);
+
+  button = gtk_button_new_with_mnemonic ("Demonstrate _Details");
+
+  g_signal_connect_swapped (button,
+                            "clicked",
+                            G_CALLBACK (demonstrate_details),
+                            &calendar_data);
+
+  gtk_box_pack_start (GTK_BOX (hbox), button, FALSE, TRUE, 0);
+
+  button = gtk_button_new_with_mnemonic ("_Reset Details");
+
+  g_signal_connect_swapped (button,
+                            "clicked",
+                            G_CALLBACK (reset_details),
+                            &calendar_data);
+
+  gtk_box_pack_start (GTK_BOX (hbox), button, FALSE, TRUE, 0);
+
+  toggle = gtk_check_button_new_with_mnemonic ("_Use Details");
+  g_signal_connect (toggle, "toggled",
+                    G_CALLBACK(calendar_toggle_details),
+                    &calendar_data);
+  gtk_box_pack_start (GTK_BOX (vbox), toggle, FALSE, TRUE, 0);
   
   /* Build the Right frame with the flags in */ 
 
-  frame = gtk_frame_new("Flags");
-  gtk_box_pack_start(GTK_BOX(vbox2), frame, TRUE, TRUE, DEF_PAD);
-  vbox3 = gtk_vbox_new(TRUE, DEF_PAD_SMALL);
-  gtk_container_add(GTK_CONTAINER(frame), vbox3);
+  vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, 0);
+  frame = create_expander ("<b>Flags</b>", vbox, 1, 0);
+  gtk_box_pack_start (GTK_BOX (rpane), frame, TRUE, TRUE, 0);
 
-  for (i = 0; i < 5; i++)
+  for (i = 0; i < G_N_ELEMENTS (calendar_data.settings); i++)
     {
-      toggle = gtk_check_button_new_with_label(flags[i].label);
-      g_signal_connect (toggle,
-                       "toggled",
-                       G_CALLBACK(calendar_toggle_flag),
+      toggle = gtk_check_button_new_with_mnemonic(flags[i].label);
+      gtk_box_pack_start (GTK_BOX (vbox), toggle, FALSE, TRUE, 0);
+      calendar_data.flag_checkboxes[i] = toggle;
+
+      g_signal_connect (toggle, "toggled",
+                        G_CALLBACK (calendar_toggle_flag),
                        &calendar_data);
-      gtk_box_pack_start (GTK_BOX (vbox3), toggle, TRUE, TRUE, 0);
-      calendar_data.flag_checkboxes[i]=toggle;
+
+      gtk_toggle_button_set_active (GTK_TOGGLE_BUTTON (toggle), flags[i].init);
     }
-  /* Build the right font-button */ 
-  button = gtk_button_new_with_label("Font...");
-  g_signal_connect (button,
-                   "clicked",
-                   G_CALLBACK(calendar_select_font),
-                   &calendar_data);
-  gtk_box_pack_start (GTK_BOX (vbox2), button, FALSE, FALSE, 0);
 
   /*
    *  Build the Signal-event part.
    */
 
-  frame = gtk_frame_new("Signal events");
-  gtk_box_pack_start(GTK_BOX(vbox), frame, TRUE, TRUE, DEF_PAD);
-
-  vbox2 = gtk_vbox_new(TRUE, DEF_PAD_SMALL);
-  gtk_container_add(GTK_CONTAINER(frame), vbox2);
+  vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, DEF_PAD_SMALL);
+  gtk_box_set_homogeneous (GTK_BOX (vbox), TRUE);
+  frame = create_frame ("<b>Signal Events</b>", vbox, 1, 0);
   
-  hbox = gtk_hbox_new (FALSE, 3);
-  gtk_box_pack_start (GTK_BOX (vbox2), hbox, FALSE, TRUE, 0);
+  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 3);
+  gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, TRUE, 0);
   label = gtk_label_new ("Signal:");
   gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, TRUE, 0);
   calendar_data.last_sig = gtk_label_new ("");
   gtk_box_pack_start (GTK_BOX (hbox), calendar_data.last_sig, FALSE, TRUE, 0);
 
-  hbox = gtk_hbox_new (FALSE, 3);
-  gtk_box_pack_start (GTK_BOX (vbox2), hbox, FALSE, TRUE, 0);
+  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 3);
+  gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, TRUE, 0);
   label = gtk_label_new ("Previous signal:");
   gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, TRUE, 0);
   calendar_data.prev_sig = gtk_label_new ("");
   gtk_box_pack_start (GTK_BOX (hbox), calendar_data.prev_sig, FALSE, TRUE, 0);
 
-  hbox = gtk_hbox_new (FALSE, 3);
-  gtk_box_pack_start (GTK_BOX (vbox2), hbox, FALSE, TRUE, 0);
+  hbox = gtk_box_new (GTK_ORIENTATION_HORIZONTAL, 3);
+  gtk_box_pack_start (GTK_BOX (vbox), hbox, FALSE, TRUE, 0);
   label = gtk_label_new ("Second previous signal:");
   gtk_box_pack_start (GTK_BOX (hbox), label, FALSE, TRUE, 0);
   calendar_data.prev2_sig = gtk_label_new ("");
   gtk_box_pack_start (GTK_BOX (hbox), calendar_data.prev2_sig, FALSE, TRUE, 0);
 
-  bbox = gtk_hbutton_box_new ();
-  gtk_box_pack_start (GTK_BOX (vbox), bbox, FALSE, FALSE, 0);
+  /*
+   *  Glue everything together
+   */
+
+  bbox = gtk_button_box_new (GTK_ORIENTATION_HORIZONTAL);
   gtk_button_box_set_layout(GTK_BUTTON_BOX(bbox), GTK_BUTTONBOX_END);
 
   button = gtk_button_new_with_label ("Close");
-  g_signal_connect (button, "clicked", 
-                   G_CALLBACK (gtk_main_quit), 
-                   NULL);
+  g_signal_connect (button, "clicked", G_CALLBACK (gtk_main_quit), NULL);
   gtk_container_add (GTK_CONTAINER (bbox), button);
-  GTK_WIDGET_SET_FLAGS (button, GTK_CAN_DEFAULT);
+
+  vbox = gtk_box_new (GTK_ORIENTATION_VERTICAL, DEF_PAD_SMALL);
+
+  gtk_box_pack_start (GTK_BOX (vbox), hpaned,
+                      TRUE,  TRUE, 0);
+  gtk_box_pack_start (GTK_BOX (vbox), gtk_separator_new (GTK_ORIENTATION_HORIZONTAL),
+                      FALSE, TRUE, 0);
+  gtk_box_pack_start (GTK_BOX (vbox), frame,
+                      FALSE, TRUE, 0);
+  gtk_box_pack_start (GTK_BOX (vbox), gtk_separator_new (GTK_ORIENTATION_HORIZONTAL),
+                      FALSE, TRUE, 0);
+  gtk_box_pack_start (GTK_BOX (vbox), bbox,
+                      FALSE, TRUE, 0);
+
+  gtk_container_add (GTK_CONTAINER (window), vbox);
+
+  gtk_widget_set_can_default (button, TRUE);
   gtk_widget_grab_default (button);
 
-  gtk_widget_show_all(window);
+  gtk_window_set_default_size (GTK_WINDOW (window), 600, 0);
+  gtk_widget_show_all (window);
 }
 
 
@@ -408,6 +679,9 @@ int main(int   argc,
 {
   gtk_init (&argc, &argv);
 
+  if (g_getenv ("GTK_RTL"))
+    gtk_widget_set_default_direction (GTK_TEXT_DIR_RTL);
+
   create_calendar();
 
   gtk_main();