Made the datasheet redraw itself when new data is loaded.
[pspp-builds.git] / lib / gtksheet / gtksheet.c
index bf6f765c316204118784bcc58878ef90464b1d45..9a4e84faa85b39f0026565eb3b511e79100830c4 100644 (file)
@@ -95,8 +95,8 @@ enum
 
 #define CELL_SPACING 1
 #define DRAG_WIDTH 6
-#define TIMEOUT_SCROLL 20
 #define TIMEOUT_FLASH 200
+#define TIMEOUT_HOVER 300
 #define TIME_INTERVAL 8
 #define COLUMN_MIN_WIDTH 10
 #define MINROWS 1
@@ -460,10 +460,18 @@ static inline gint SHEET_WIDTH (GtkSheet *sheet)
   return cx;
 }
 
-#define MIN_VISIBLE_ROW(sheet) sheet->view.row0
-#define MAX_VISIBLE_ROW(sheet) sheet->view.rowi
-#define MIN_VISIBLE_COLUMN(sheet) sheet->view.col0
-#define MAX_VISIBLE_COLUMN(sheet) sheet->view.coli
+#define MIN_VISIBLE_ROW(sheet) \
+    ROW_FROM_YPIXEL (sheet, sheet->column_title_area.height + 1)
+
+#define MAX_VISIBLE_ROW(sheet) \
+    ROW_FROM_YPIXEL (sheet, sheet->sheet_window_height - 1)
+
+#define MIN_VISIBLE_COLUMN(sheet) \
+    COLUMN_FROM_XPIXEL (sheet, sheet->row_title_area.width + 1)
+
+#define MAX_VISIBLE_COLUMN(sheet) \
+    COLUMN_FROM_XPIXEL (sheet, sheet->sheet_window_width)
+
 
 
 static inline gboolean
@@ -586,10 +594,10 @@ POSSIBLE_RESIZE (const GtkSheet *sheet, gint x, gint y,
     yyy_row_height (sheet, sheet->range.rowi);
 
   if (sheet->state == GTK_SHEET_COLUMN_SELECTED)
-    ydrag = ROW_TOP_YPIXEL (sheet, sheet->view.row0);
+    ydrag = ROW_TOP_YPIXEL (sheet, MIN_VISIBLE_ROW (sheet));
 
   if (sheet->state == GTK_SHEET_ROW_SELECTED)
-    xdrag = COLUMN_LEFT_XPIXEL (sheet, sheet->view.col0);
+    xdrag = COLUMN_LEFT_XPIXEL (sheet, MIN_VISIBLE_COLUMN (sheet));
 
   *drag_column = COLUMN_FROM_XPIXEL (sheet,x);
   *drag_row = ROW_FROM_YPIXEL (sheet,y);
@@ -644,7 +652,6 @@ static gint gtk_sheet_cell_isvisible                 (GtkSheet * sheet,
                                                  gint row, gint column);
 /* Clipped Range */
 
-static gint gtk_sheet_scroll                    (gpointer data);
 static gint gtk_sheet_flash                     (gpointer data);
 
 /* Drawing Routines */
@@ -667,7 +674,7 @@ static void gtk_sheet_range_draw_selection   (GtkSheet *sheet,
 
 /* Selection */
 
-static gint gtk_sheet_move_query                (GtkSheet *sheet,
+static gboolean gtk_sheet_move_query                    (GtkSheet *sheet,
                                                  gint row, gint column);
 static void gtk_sheet_real_select_range         (GtkSheet * sheet,
                                                  const GtkSheetRange * range);
@@ -707,10 +714,6 @@ static void gtk_sheet_draw_backing_pixmap   (GtkSheet *sheet,
 /* Scrollbars */
 
 static void adjust_scrollbars                   (GtkSheet * sheet);
-static void vadjustment_changed                         (GtkAdjustment * adjustment,
-                                                 gpointer data);
-static void hadjustment_changed                 (GtkAdjustment * adjustment,
-                                                 gpointer data);
 static void vadjustment_value_changed           (GtkAdjustment * adjustment,
                                                  gpointer data);
 static void hadjustment_value_changed           (GtkAdjustment * adjustment,
@@ -1112,11 +1115,6 @@ gtk_sheet_init (GtkSheet *sheet)
   GTK_WIDGET_UNSET_FLAGS (sheet, GTK_NO_WINDOW);
   GTK_WIDGET_SET_FLAGS (sheet, GTK_CAN_FOCUS);
 
-  sheet->view.row0 = 0;
-  sheet->view.col0 = 0;
-  sheet->view.rowi = 0;
-  sheet->view.coli = 0;
-
   sheet->column_title_window = NULL;
   sheet->column_title_area.x = 0;
   sheet->column_title_area.y = 0;
@@ -1169,6 +1167,8 @@ gtk_sheet_init (GtkSheet *sheet)
   gdk_color_parse ("gray", &sheet->grid_color);
   gdk_color_alloc (gdk_colormap_get_system (), &sheet->grid_color);
   sheet->show_grid = TRUE;
+
+  sheet->motion_events = 0;
 }
 
 
@@ -1193,15 +1193,6 @@ columns_inserted_deleted_callback (GSheetModel *model, gint first_column,
   range.coli = xxx_column_count (sheet) - 1;
   range.rowi = yyy_row_count (sheet) - 1;
 
-  sheet->view.col0 =
-    COLUMN_FROM_XPIXEL (sheet, sheet->row_title_area.width + 1);
-
-  sheet->view.coli =
-    COLUMN_FROM_XPIXEL (sheet, sheet->sheet_window_width);
-
-  if ( sheet->view.coli > range.coli)
-    sheet->view.coli = range.coli;
-
   adjust_scrollbars (sheet);
 
   if (sheet->active_cell.col >= model_columns)
@@ -1234,14 +1225,6 @@ rows_inserted_deleted_callback (GSheetModel *model, gint first_row,
   range.rowi = yyy_row_count (sheet) - 1;
   range.coli = xxx_column_count (sheet) - 1;
 
-  sheet->view.row0 =
-    ROW_FROM_YPIXEL (sheet, sheet->column_title_area.height + 1);
-  sheet->view.rowi =
-    ROW_FROM_YPIXEL (sheet, sheet->sheet_window_height - 1);
-
-  if ( sheet->view.rowi > range.rowi)
-    sheet->view.rowi = range.rowi;
-
   adjust_scrollbars (sheet);
 
   if (sheet->active_cell.row >= model_rows)
@@ -1272,7 +1255,17 @@ range_update_callback (GSheetModel *m, gint row0, gint col0,
 
   if ( ( row0 < 0 && col0 < 0 ) || ( rowi < 0 && coli < 0 ) )
     {
+      gint i;
       gtk_sheet_range_draw (sheet, NULL);
+      adjust_scrollbars (sheet);
+
+      for (i = MIN_VISIBLE_ROW (sheet); i <= MAX_VISIBLE_ROW (sheet); i++)
+       gtk_sheet_row_title_button_draw (sheet, i);
+
+      for (i = MIN_VISIBLE_COLUMN (sheet);
+          i <= MAX_VISIBLE_COLUMN (sheet); i++)
+       gtk_sheet_column_title_button_draw (sheet, i);
+
       return;
     }
   else if ( row0 < 0 || rowi < 0 )
@@ -1810,9 +1803,6 @@ gtk_sheet_set_row_titles_width (GtkSheet *sheet, guint width)
   if (width < COLUMN_MIN_WIDTH) return;
 
   sheet->row_title_area.width = width;
-  sheet->view.col0 = COLUMN_FROM_XPIXEL (sheet, sheet->row_title_area.width + 1);
-  sheet->view.coli = COLUMN_FROM_XPIXEL (sheet, sheet->sheet_window_width);
-
 
   adjust_scrollbars (sheet);
 
@@ -1829,9 +1819,6 @@ gtk_sheet_set_column_titles_height (GtkSheet *sheet, guint height)
   if (height < DEFAULT_ROW_HEIGHT (GTK_WIDGET (sheet))) return;
 
   sheet->column_title_area.height = height;
-  sheet->view.row0 = ROW_FROM_YPIXEL (sheet,
-                                     sheet->column_title_area.height + 1);
-  sheet->view.rowi = ROW_FROM_YPIXEL (sheet, sheet->sheet_window_height - 1);
 
   adjust_scrollbars (sheet);
 
@@ -2010,7 +1997,7 @@ gtk_sheet_row_titles_visible (GtkSheet *sheet)
 }
 
 void
-gtk_sheet_moveto (GtkSheet * sheet,
+gtk_sheet_moveto (GtkSheet *sheet,
                  gint row,
                  gint column,
                  gfloat row_align,
@@ -2214,6 +2201,8 @@ gtk_sheet_select_column (GtkSheet * sheet, gint column)
   gtk_sheet_real_select_range (sheet, NULL);
 }
 
+
+
 void
 gtk_sheet_clip_range (GtkSheet *sheet, const GtkSheetRange *range)
 {
@@ -2248,6 +2237,7 @@ gtk_sheet_unclip_range (GtkSheet *sheet)
   if (!GTK_SHEET_IN_CLIP (sheet)) return;
 
   GTK_SHEET_UNSET_FLAGS (sheet, GTK_SHEET_IN_CLIP);
+
   gtk_timeout_remove (sheet->clip_timer);
   gtk_sheet_range_draw (sheet, &sheet->clip_range);
 
@@ -2443,7 +2433,6 @@ gtk_sheet_cell_isvisible (GtkSheet * sheet,
 void
 gtk_sheet_get_visible_range (GtkSheet *sheet, GtkSheetRange *range)
 {
-
   g_return_if_fail (sheet != NULL);
   g_return_if_fail (GTK_IS_SHEET (sheet)) ;
   g_return_if_fail (range != NULL);
@@ -2452,7 +2441,6 @@ gtk_sheet_get_visible_range (GtkSheet *sheet, GtkSheetRange *range)
   range->col0 = MIN_VISIBLE_COLUMN (sheet);
   range->rowi = MAX_VISIBLE_ROW (sheet);
   range->coli = MAX_VISIBLE_COLUMN (sheet);
-
 }
 
 GtkAdjustment *
@@ -2502,9 +2490,6 @@ gtk_sheet_set_vadjustment (GtkSheet *sheet,
       gtk_object_ref (GTK_OBJECT (sheet->vadjustment));
       gtk_object_sink (GTK_OBJECT (sheet->vadjustment));
 
-      gtk_signal_connect (GTK_OBJECT (sheet->vadjustment), "changed",
-                         (GtkSignalFunc) vadjustment_changed,
-                         (gpointer) sheet);
       gtk_signal_connect (GTK_OBJECT (sheet->vadjustment), "value_changed",
                          (GtkSignalFunc) vadjustment_value_changed,
                          (gpointer) sheet);
@@ -2548,9 +2533,6 @@ gtk_sheet_set_hadjustment (GtkSheet *sheet,
       gtk_object_ref (GTK_OBJECT (sheet->hadjustment));
       gtk_object_sink (GTK_OBJECT (sheet->hadjustment));
 
-      gtk_signal_connect (GTK_OBJECT (sheet->hadjustment), "changed",
-                         (GtkSignalFunc) hadjustment_changed,
-                         (gpointer) sheet);
       gtk_signal_connect (GTK_OBJECT (sheet->hadjustment), "value_changed",
                          (GtkSignalFunc) hadjustment_value_changed,
                          (gpointer) sheet);
@@ -2626,12 +2608,6 @@ gtk_sheet_destroy (GtkObject * object)
       sheet->button = NULL;
     }
 
-  if (sheet->timer)
-    {
-      gtk_timeout_remove (sheet->timer);
-      sheet->timer = 0;
-    }
-
   if (sheet->clip_timer)
     {
       gtk_timeout_remove (sheet->clip_timer);
@@ -3310,7 +3286,8 @@ gtk_sheet_range_draw (GtkSheet *sheet, const GtkSheetRange *range)
     {
       drawing_range.row0 = MIN_VISIBLE_ROW (sheet);
       drawing_range.col0 = MIN_VISIBLE_COLUMN (sheet);
-      drawing_range.rowi = MIN (MAX_VISIBLE_ROW (sheet), yyy_row_count (sheet) - 1);
+      drawing_range.rowi = MIN (MAX_VISIBLE_ROW (sheet),
+                               yyy_row_count (sheet) - 1);
       drawing_range.coli = MAX_VISIBLE_COLUMN (sheet);
 
 
@@ -3329,7 +3306,6 @@ gtk_sheet_range_draw (GtkSheet *sheet, const GtkSheetRange *range)
       drawing_range.coli = MIN (range->coli, MAX_VISIBLE_COLUMN (sheet));
     }
 
-
   if (drawing_range.coli == xxx_column_count (sheet) - 1)
     {
       area.x = COLUMN_LEFT_XPIXEL (sheet,
@@ -3459,7 +3435,7 @@ gtk_sheet_range_draw_selection (GtkSheet *sheet, GtkSheetRange range)
                }
              if (j == sheet->range.coli) area.width = area.width - 3;
 
-             if (i!=sheet->active_cell.row || j!=sheet->active_cell.col)
+             if (i != sheet->active_cell.row || j != sheet->active_cell.col)
                {
                  gdk_draw_rectangle (sheet->sheet_window,
                                      sheet->xor_gc,
@@ -3603,8 +3579,8 @@ gtk_sheet_set_cell (GtkSheet *sheet, gint row, gint col,
 
       range.row0 = row;
       range.rowi = row;
-      range.col0 = sheet->view.col0;
-      range.coli = sheet->view.coli;
+      range.col0 = MIN_VISIBLE_COLUMN (sheet);
+      range.coli = MAX_VISIBLE_COLUMN (sheet);
 
       if (gtk_sheet_autoresize (sheet) &&
          text_width > xxx_column_width (sheet, col) -
@@ -3639,8 +3615,8 @@ gtk_sheet_cell_clear (GtkSheet *sheet, gint row, gint column)
 
   range.row0 = row;
   range.rowi = row;
-  range.col0 = sheet->view.col0;
-  range.coli = sheet->view.coli;
+  range.col0 = MIN_VISIBLE_COLUMN (sheet);
+  range.coli = MAX_VISIBLE_COLUMN (sheet);
 
   gtk_sheet_real_cell_clear (sheet, row, column, FALSE);
 
@@ -3662,8 +3638,8 @@ gtk_sheet_cell_delete (GtkSheet *sheet, gint row, gint column)
 
   range.row0 = row;
   range.rowi = row;
-  range.col0 = sheet->view.col0;
-  range.coli = sheet->view.coli;
+  range.col0 = MIN_VISIBLE_COLUMN (sheet);
+  range.coli = MAX_VISIBLE_COLUMN (sheet);
 
   gtk_sheet_real_cell_clear (sheet, row, column, TRUE);
 
@@ -3812,31 +3788,54 @@ gtk_sheet_cell_get_state (GtkSheet *sheet, gint row, gint col)
   return GTK_STATE_NORMAL;
 }
 
+/* Convert X, Y (in pixels) to *ROW, *COLUMN (in cell coords)
+   -1 indicates the title buttons.
+   If the function returns FALSE, then the results will be unreliable.
+*/
 gboolean
-gtk_sheet_get_pixel_info (GtkSheet * sheet,
+gtk_sheet_get_pixel_info (GtkSheet *sheet,
                          gint x,
                          gint y,
-                         gint * row,
-                         gint * column)
+                         gint *row,
+                         gint *column)
 {
   gint trow, tcol;
+  *row = -G_MAXINT;
+  *column = -G_MAXINT;
 
   g_return_val_if_fail (sheet != NULL, 0);
   g_return_val_if_fail (GTK_IS_SHEET (sheet), 0);
 
   /* bounds checking, return false if the user clicked
      on a blank area */
-  trow = ROW_FROM_YPIXEL (sheet, y);
-  if (trow >= yyy_row_count (sheet))
+  if (y < 0)
     return FALSE;
 
-  *row = trow;
-
-  tcol = COLUMN_FROM_XPIXEL (sheet, x);
-  if (tcol >= xxx_column_count (sheet))
+  if (x < 0)
     return FALSE;
 
-  *column = tcol;
+  if ( y < sheet->column_title_area.height + sheet->column_title_area.y)
+    *row = -1;
+
+  else
+    {
+      trow = ROW_FROM_YPIXEL (sheet, y);
+      if (trow > yyy_row_count (sheet))
+       return FALSE;
+
+      *row = trow;
+    }
+
+  if ( x < sheet->row_title_area.width + sheet->row_title_area.x)
+    *column = -1;
+  else
+    {
+      tcol = COLUMN_FROM_XPIXEL (sheet, x);
+      if (tcol > xxx_column_count (sheet))
+       return FALSE;
+
+      *column = tcol;
+    }
 
   return TRUE;
 }
@@ -3908,8 +3907,8 @@ gtk_sheet_get_active_cell (GtkSheet *sheet, gint *row, gint *column)
   g_return_if_fail (sheet != NULL);
   g_return_if_fail (GTK_IS_SHEET (sheet));
 
-  *row = sheet->active_cell.row;
-  *column = sheet->active_cell.col;
+  if ( row ) *row = sheet->active_cell.row;
+  if (column) *column = sheet->active_cell.col;
 }
 
 static void
@@ -4391,7 +4390,7 @@ gtk_sheet_new_selection (GtkSheet *sheet, GtkSheetRange *range)
          selected= (i <= new_range.rowi && i >= new_range.row0 &&
                     j <= new_range.coli && j >= new_range.col0) ? TRUE : FALSE;
 
-         if (state!=GTK_STATE_SELECTED && selected &&
+         if (state != GTK_STATE_SELECTED && selected &&
              xxx_column_is_visible (sheet, j) && yyy_row_is_visible (sheet, i) &&
              (i != sheet->active_cell.row || j != sheet->active_cell.col))
            {
@@ -4444,7 +4443,7 @@ gtk_sheet_new_selection (GtkSheet *sheet, GtkSheetRange *range)
              mask2 = i == new_range.rowi ? mask2 + 2 : mask2;
              mask2 = j == new_range.col0 ? mask2 + 4 : mask2;
              mask2 = j == new_range.coli ? mask2 + 8 : mask2;
-             if (mask2!=mask1 || (mask2 == mask1 && state!=GTK_STATE_SELECTED))
+             if (mask2 != mask1 || (mask2 == mask1 && state != GTK_STATE_SELECTED))
                {
                  x = COLUMN_LEFT_XPIXEL (sheet,j);
                  y = ROW_TOP_YPIXEL (sheet, i);
@@ -4583,7 +4582,7 @@ gtk_sheet_draw_corners (GtkSheet *sheet, GtkSheetRange range)
       width = 1;
       if (sheet->state == GTK_SHEET_COLUMN_SELECTED)
        {
-         y = ROW_TOP_YPIXEL (sheet, sheet->view.row0)+3;
+         y = ROW_TOP_YPIXEL (sheet, MIN_VISIBLE_ROW (sheet))+3;
          width = 3;
        }
       gdk_draw_pixmap (sheet->sheet_window,
@@ -4611,7 +4610,7 @@ gtk_sheet_draw_corners (GtkSheet *sheet, GtkSheetRange range)
       width = 1;
       if (sheet->state == GTK_SHEET_ROW_SELECTED)
        {
-         x = COLUMN_LEFT_XPIXEL (sheet, sheet->view.col0)+3;
+         x = COLUMN_LEFT_XPIXEL (sheet, MIN_VISIBLE_COLUMN (sheet))+3;
          width = 3;
        }
       gdk_draw_pixmap (sheet->sheet_window,
@@ -4690,6 +4689,16 @@ gtk_sheet_real_select_range (GtkSheet * sheet,
   gtk_signal_emit (GTK_OBJECT (sheet), sheet_signals[SELECT_RANGE], &sheet->range);
 }
 
+
+void
+gtk_sheet_get_selected_range           (GtkSheet *sheet,
+                                        GtkSheetRange *range)
+{
+  g_return_if_fail (sheet != NULL);
+  *range = sheet->range;
+}
+
+
 void
 gtk_sheet_select_range (GtkSheet * sheet, const GtkSheetRange *range)
 {
@@ -4837,7 +4846,7 @@ gtk_sheet_expose (GtkWidget * widget,
 }
 
 
-static gint
+static gboolean
 gtk_sheet_button_press (GtkWidget * widget,
                        GdkEventButton * event)
 {
@@ -4936,7 +4945,11 @@ gtk_sheet_button_press (GtkWidget * widget,
                        GDK_BUTTON_RELEASE_MASK,
                        NULL, NULL, event->time);
       gtk_grab_add (GTK_WIDGET (sheet));
-      sheet->timer = gtk_timeout_add (TIMEOUT_SCROLL, gtk_sheet_scroll, sheet);
+
+      /* This seems to be a kludge to work around a problem where the sheet
+        scrolls to another position.  The timeout scrolls it back to its
+        original posn.          JMD 3 July 2007
+      */
       gtk_widget_grab_focus (GTK_WIDGET (sheet));
 
       if (sheet->selection_mode != GTK_SELECTION_SINGLE &&
@@ -5007,11 +5020,11 @@ gtk_sheet_button_press (GtkWidget * widget,
     {
       gtk_widget_get_pointer (widget, &x, &y);
       column = COLUMN_FROM_XPIXEL (sheet, x);
+
       if (xxx_column_is_sensitive (sheet, column))
        {
          gtk_sheet_click_cell (sheet, - 1, column, &veto);
          gtk_grab_add (GTK_WIDGET (sheet));
-         sheet->timer = gtk_timeout_add (TIMEOUT_SCROLL, gtk_sheet_scroll, sheet);
          gtk_widget_grab_focus (GTK_WIDGET (sheet));
          GTK_SHEET_SET_FLAGS (sheet, GTK_SHEET_IN_SELECTION);
        }
@@ -5025,7 +5038,6 @@ gtk_sheet_button_press (GtkWidget * widget,
        {
          gtk_sheet_click_cell (sheet, row, - 1, &veto);
          gtk_grab_add (GTK_WIDGET (sheet));
-         sheet->timer = gtk_timeout_add (TIMEOUT_SCROLL, gtk_sheet_scroll, sheet);
          gtk_widget_grab_focus (GTK_WIDGET (sheet));
          GTK_SHEET_SET_FLAGS (sheet, GTK_SHEET_IN_SELECTION);
        }
@@ -5034,6 +5046,7 @@ gtk_sheet_button_press (GtkWidget * widget,
   return TRUE;
 }
 
+#if 0
 static gint
 gtk_sheet_scroll (gpointer data)
 {
@@ -5062,8 +5075,8 @@ gtk_sheet_scroll (gpointer data)
   GDK_THREADS_LEAVE ();
 
   return TRUE;
-
 }
+#endif
 
 static void
 gtk_sheet_click_cell (GtkSheet *sheet, gint row, gint column, gboolean *veto)
@@ -5268,8 +5281,6 @@ gtk_sheet_button_release (GtkWidget * widget,
 
   if (GTK_SHEET_IN_SELECTION)
     gdk_pointer_ungrab (event->time);
-  if (sheet->timer)
-    gtk_timeout_remove (sheet->timer);
   gtk_grab_remove (GTK_WIDGET (sheet));
 
   GTK_SHEET_UNSET_FLAGS (sheet, GTK_SHEET_IN_SELECTION);
@@ -5277,6 +5288,161 @@ gtk_sheet_button_release (GtkWidget * widget,
   return TRUE;
 }
 
+/* Shamelessly lifted from gtktooltips */
+static gboolean
+gtk_sheet_subtitle_paint_window (GtkWidget *tip_window)
+{
+  GtkRequisition req;
+
+  gtk_widget_size_request (tip_window, &req);
+  gtk_paint_flat_box (tip_window->style, tip_window->window,
+                     GTK_STATE_NORMAL, GTK_SHADOW_OUT,
+                     NULL, GTK_WIDGET(tip_window), "tooltip",
+                     0, 0, req.width, req.height);
+
+  return FALSE;
+}
+
+static GtkSheetHoverTitle *
+create_hover_window (void)
+{
+  GtkSheetHoverTitle *hw = malloc (sizeof (*hw));
+
+  hw->window = gtk_window_new (GTK_WINDOW_POPUP);
+
+#if GTK_CHECK_VERSION (2, 9, 0)
+  gtk_window_set_type_hint (GTK_WINDOW (hw->window),
+                           GDK_WINDOW_TYPE_HINT_TOOLTIP);
+#endif
+
+  gtk_widget_set_app_paintable (hw->window, TRUE);
+  gtk_window_set_resizable (GTK_WINDOW (hw->window), FALSE);
+  gtk_widget_set_name (hw->window, "gtk-tooltips");
+  gtk_container_set_border_width (GTK_CONTAINER (hw->window), 4);
+
+  g_signal_connect (hw->window,
+                   "expose_event",
+                   G_CALLBACK (gtk_sheet_subtitle_paint_window),
+                   NULL);
+
+  hw->label = gtk_label_new (NULL);
+
+
+  gtk_label_set_line_wrap (GTK_LABEL (hw->label), TRUE);
+  gtk_misc_set_alignment (GTK_MISC (hw->label), 0.5, 0.5);
+
+  gtk_container_add (GTK_CONTAINER (hw->window), hw->label);
+
+  gtk_widget_show (hw->label);
+
+  g_signal_connect (hw->window,
+                   "destroy",
+                   G_CALLBACK (gtk_widget_destroyed),
+                   &hw->window);
+
+  return hw;
+}
+
+#define HOVER_WINDOW_Y_OFFSET 2
+
+static void
+show_subtitle (GtkSheet *sheet, gint row, gint column, const gchar *subtitle)
+{
+  gint x, y;
+  gint px, py;
+  gint width;
+
+  if ( ! subtitle )
+    return;
+
+  if ( ! sheet->hover_window)
+    {
+      sheet->hover_window = create_hover_window ();
+      gtk_widget_add_events (GTK_WIDGET (sheet), GDK_LEAVE_NOTIFY_MASK);
+
+      g_signal_connect_swapped (sheet, "leave-notify-event",
+                               G_CALLBACK (gtk_widget_hide),
+                               sheet->hover_window->window);
+    }
+
+  gtk_label_set_text (GTK_LABEL (sheet->hover_window->label),
+                     subtitle);
+
+
+  sheet->hover_window->row = row;
+  sheet->hover_window->column = column;
+
+  gdk_window_get_origin (GTK_WIDGET (sheet)->window, &x, &y);
+
+  gtk_widget_get_pointer (GTK_WIDGET (sheet), &px, &py);
+
+  gtk_widget_show (sheet->hover_window->window);
+
+  width = GTK_WIDGET (sheet->hover_window->label)->allocation.width;
+
+  if (row == -1 )
+    {
+      x += px;
+      x -= width / 2;
+      y += sheet->column_title_area.y;
+      y += sheet->column_title_area.height;
+      y += HOVER_WINDOW_Y_OFFSET;
+    }
+
+  if ( column == -1 )
+    {
+      y += py;
+      x += sheet->row_title_area.x;
+      x += sheet->row_title_area.width * 2 / 3.0;
+    }
+
+  gtk_window_move (GTK_WINDOW (sheet->hover_window->window),
+                  x, y);
+}
+
+static gboolean
+motion_timeout_callback (gpointer data)
+{
+  GtkSheet *sheet = GTK_SHEET (data);
+  if ( --sheet->motion_events == 0 )
+    {
+      gint x, y;
+      gint row, column;
+      gtk_widget_get_pointer (GTK_WIDGET (sheet), &x, &y);
+
+      if ( gtk_sheet_get_pixel_info (sheet, x, y, &row, &column) )
+       {
+         if ( column == -1 && row == -1 )
+           return FALSE;
+
+         if ( column == -1)
+           {
+             GSheetRow *row_geo = sheet->row_geometry;
+             gchar *text;
+
+             text = g_sheet_row_get_subtitle (row_geo, row);
+
+             show_subtitle (sheet, row, column, text);
+             g_free (text);
+           }
+
+         if ( row == -1)
+           {
+             GSheetColumn *col_geo = sheet->column_geometry;
+             gchar *text;
+
+             text = g_sheet_column_get_subtitle (col_geo, column);
+
+             show_subtitle (sheet, row, column, text );
+
+             g_free (text);
+           }
+       }
+    }
+
+  return FALSE;
+}
+
 static gint
 gtk_sheet_motion (GtkWidget * widget,
                  GdkEventMotion * event)
@@ -5297,6 +5463,26 @@ gtk_sheet_motion (GtkWidget * widget,
   x = event->x;
   y = event->y;
 
+  if (!sheet->hover_window || ! GTK_WIDGET_VISIBLE (sheet->hover_window->window))
+    {
+      sheet->motion_events++;
+      g_timeout_add (TIMEOUT_HOVER, motion_timeout_callback, sheet);
+    }
+  else
+    {
+      gint row, column;
+      gint wx, wy;
+      gtk_widget_get_pointer (widget, &wx, &wy);
+
+      if ( gtk_sheet_get_pixel_info (sheet, wx, wy, &row, &column) )
+       {
+         if ( row != sheet->hover_window->row || column != sheet->hover_window->column)
+           {
+             gtk_widget_hide (sheet->hover_window->window);
+           }
+       }
+    }
+
   if (event->window == sheet->column_title_window &&
       gtk_sheet_columns_resizable (sheet))
     {
@@ -5473,7 +5659,8 @@ gtk_sheet_motion (GtkWidget * widget,
 
       /*use half of column width resp. row height as threshold to
        expand selection*/
-      col_threshold = COLUMN_LEFT_XPIXEL (sheet,current_col)+xxx_column_width (sheet,current_col)/2;
+      col_threshold = COLUMN_LEFT_XPIXEL (sheet,current_col) +
+       xxx_column_width (sheet,current_col) / 2;
       if (column > 0)
        {
          if (x < col_threshold)
@@ -5511,7 +5698,6 @@ gtk_sheet_motion (GtkWidget * widget,
       if (aux.row0 + row >= 0 && aux.rowi + row < yyy_row_count (sheet) &&
          aux.col0 + column >= 0 && aux.coli + column < xxx_column_count (sheet))
        {
-
          aux = sheet->drag_range;
          sheet->drag_range = sheet->range;
 
@@ -5532,8 +5718,6 @@ gtk_sheet_motion (GtkWidget * widget,
       return TRUE;
     }
 
-
-
   gtk_sheet_get_pixel_info (sheet, x, y, &row, &column);
 
   if (sheet->state == GTK_SHEET_NORMAL && row == sheet->active_cell.row &&
@@ -5545,7 +5729,7 @@ gtk_sheet_motion (GtkWidget * widget,
   return TRUE;
 }
 
-static gint
+static gboolean
 gtk_sheet_move_query (GtkSheet *sheet, gint row, gint column)
 {
   gint row_move, column_move;
@@ -5562,7 +5746,7 @@ gtk_sheet_move_query (GtkSheet *sheet, gint row, gint column)
   height = sheet->sheet_window_height;
   width = sheet->sheet_window_width;
 
-  if (row >= MAX_VISIBLE_ROW (sheet) && sheet->state!=GTK_SHEET_COLUMN_SELECTED)
+  if (row >= MAX_VISIBLE_ROW (sheet) && sheet->state != GTK_SHEET_COLUMN_SELECTED)
     {
       row_align = 1.;
       new_row = MIN (yyy_row_count (sheet), row + 1);
@@ -5575,12 +5759,12 @@ gtk_sheet_move_query (GtkSheet *sheet, gint row, gint column)
          row_align = -1.;
        }
     }
-  if (row < MIN_VISIBLE_ROW (sheet) && sheet->state!=GTK_SHEET_COLUMN_SELECTED)
+  if (row < MIN_VISIBLE_ROW (sheet) && sheet->state != GTK_SHEET_COLUMN_SELECTED)
     {
       row_align= 0.;
       row_move = TRUE;
     }
-  if (column >= MAX_VISIBLE_COLUMN (sheet) && sheet->state!=GTK_SHEET_ROW_SELECTED)
+  if (column >= MAX_VISIBLE_COLUMN (sheet) && sheet->state != GTK_SHEET_ROW_SELECTED)
     {
       col_align = 1.;
       new_col = MIN (xxx_column_count (sheet) - 1, column + 1);
@@ -5593,7 +5777,7 @@ gtk_sheet_move_query (GtkSheet *sheet, gint row, gint column)
          col_align = -1.;
        }
     }
-  if (column < MIN_VISIBLE_COLUMN (sheet) && sheet->state!=GTK_SHEET_ROW_SELECTED)
+  if (column < MIN_VISIBLE_COLUMN (sheet) && sheet->state != GTK_SHEET_ROW_SELECTED)
     {
       col_align = 0.;
       column_move = TRUE;
@@ -5963,17 +6147,6 @@ gtk_sheet_size_request (GtkWidget * widget,
   if (sheet->row_titles_visible)
     requisition->width += sheet->row_title_area.width;
 
-  sheet->view.row0 = ROW_FROM_YPIXEL (sheet, sheet->column_title_area.height + 1);
-  sheet->view.rowi = ROW_FROM_YPIXEL (sheet, sheet->sheet_window_height - 1);
-  sheet->view.col0 = COLUMN_FROM_XPIXEL (sheet, sheet->row_title_area.width + 1);
-  sheet->view.coli = COLUMN_FROM_XPIXEL (sheet, sheet->sheet_window_width);
-
-  if (!sheet->column_titles_visible)
-    sheet->view.row0 = ROW_FROM_YPIXEL (sheet, 1);
-
-  if (!sheet->row_titles_visible)
-    sheet->view.col0 = COLUMN_FROM_XPIXEL (sheet, 1);
-
   children = sheet->children;
   while (children)
     {
@@ -6069,20 +6242,7 @@ gtk_sheet_size_allocate (GtkWidget * widget,
 
   /* row button allocation */
   size_allocate_row_title_buttons (sheet);
-
-  sheet->view.row0 = ROW_FROM_YPIXEL (sheet, sheet->column_title_area.height + 1);
-  sheet->view.rowi = ROW_FROM_YPIXEL (sheet, sheet->sheet_window_height - 1);
-  sheet->view.col0 = COLUMN_FROM_XPIXEL (sheet, sheet->row_title_area.width + 1);
-  sheet->view.coli = COLUMN_FROM_XPIXEL (sheet, sheet->sheet_window_width);
-
-  if (!sheet->column_titles_visible)
-    sheet->view.row0 = ROW_FROM_YPIXEL (sheet, 1);
-
-  if (!sheet->row_titles_visible)
-    sheet->view.col0 = COLUMN_FROM_XPIXEL (sheet, 1);
-
   size_allocate_column_title_buttons (sheet);
-  size_allocate_row_title_buttons (sheet);
 
   /* re - scale backing pixmap */
   gtk_sheet_make_backing_pixmap (sheet, 0, 0);
@@ -6753,15 +6913,12 @@ gtk_sheet_row_title_button_draw (GtkSheet *sheet, gint row)
  *
  * functions:
  * adjust_scrollbars
- * vadjustment_changed
- * hadjustment_changed
  * vadjustment_value_changed
  * hadjustment_value_changed */
 
 static void
 adjust_scrollbars (GtkSheet * sheet)
 {
-
   if (sheet->vadjustment)
     {
       sheet->vadjustment->page_size = sheet->sheet_window_height;
@@ -6785,33 +6942,6 @@ adjust_scrollbars (GtkSheet * sheet)
     }
 }
 
-
-static void
-vadjustment_changed (GtkAdjustment * adjustment,
-                    gpointer data)
-{
-  GtkSheet *sheet;
-
-  g_return_if_fail (adjustment != NULL);
-  g_return_if_fail (data != NULL);
-
-  sheet = GTK_SHEET (data);
-
-}
-
-static void
-hadjustment_changed (GtkAdjustment * adjustment,
-                    gpointer data)
-{
-  GtkSheet *sheet;
-
-  g_return_if_fail (adjustment != NULL);
-  g_return_if_fail (data != NULL);
-
-  sheet = GTK_SHEET (data);
-}
-
-
 static void
 vadjustment_value_changed (GtkAdjustment * adjustment,
                           gpointer data)
@@ -6893,11 +7023,6 @@ vadjustment_value_changed (GtkAdjustment * adjustment,
 
   sheet->voffset = - value;
 
-  sheet->view.row0 = ROW_FROM_YPIXEL (sheet, sheet->column_title_area.height + 1);
-  sheet->view.rowi = ROW_FROM_YPIXEL (sheet, sheet->sheet_window_height - 1);
-  if (!sheet->column_titles_visible)
-    sheet->view.row0 = ROW_FROM_YPIXEL (sheet, 1);
-
   if (GTK_WIDGET_REALIZED (sheet->sheet_entry) &&
       sheet->state == GTK_SHEET_NORMAL &&
       sheet->active_cell.row >= 0 && sheet->active_cell.col >= 0 &&
@@ -7004,12 +7129,6 @@ hadjustment_value_changed (GtkAdjustment * adjustment,
     }
 
   sheet->hoffset = - value;
-
-  sheet->view.col0 = COLUMN_FROM_XPIXEL (sheet, sheet->row_title_area.width + 1);
-  sheet->view.coli = COLUMN_FROM_XPIXEL (sheet, sheet->sheet_window_width);
-  if (!sheet->row_titles_visible)
-    sheet->view.col0 = COLUMN_FROM_XPIXEL (sheet, 1);
-
   if (GTK_WIDGET_REALIZED (sheet->sheet_entry) &&
       sheet->state == GTK_SHEET_NORMAL &&
       sheet->active_cell.row >= 0 && sheet->active_cell.col >= 0 &&
@@ -7157,7 +7276,6 @@ new_column_width (GtkSheet * sheet,
     width = min_width;
 
   xxx_set_column_width (sheet, column, width);
-  sheet->view.coli = COLUMN_FROM_XPIXEL (sheet, sheet->sheet_window_width);
   size_allocate_column_title_buttons (sheet);
 
   return width;
@@ -7191,7 +7309,6 @@ new_row_height (GtkSheet * sheet,
     height = min_height;
 
   yyy_set_row_height (sheet, row, height);
-  sheet->view.rowi = ROW_FROM_YPIXEL (sheet, sheet->sheet_window_height - 1);
   size_allocate_row_title_buttons (sheet);
 
   return height;
@@ -7430,7 +7547,8 @@ gtk_sheet_attach_default (GtkSheet *sheet,
       return;
     }
 
-  gtk_sheet_attach (sheet, widget, row, col, GTK_EXPAND|GTK_FILL, GTK_EXPAND|GTK_FILL, 0, 0);
+  gtk_sheet_attach (sheet, widget, row, col,
+                   GTK_EXPAND | GTK_FILL, GTK_EXPAND | GTK_FILL, 0, 0);
 }
 
 void